XSLT - эффективный способ проверить, содержит ли поле в повторяющейся записи другое значение поля? - PullRequest
0 голосов
/ 07 мая 2019

Я работаю над проектом в BizTalk, где я получаю HR-данные, которые, как ожидается, будут преобразованы в другой формат, прежде чем они будут направлены в пункт назначения.

Вот пример того, что я получаю:

<ns0:Employees xmlns:ns0="http://XXX">
  <Employee>
    <EMPLOYEE_ID>1</EMPLOYEE_ID>
    <BOSS_ID>2</BOSS_ID>
  </Employee>
  <Employee>
    <EMPLOYEE_ID>2</EMPLOYEE_ID>
    <BOSS_ID>3</BOSS_ID>
  </Employee>
  <Employee>
    <EMPLOYEE_ID>3</EMPLOYEE_ID>
    <BOSS_ID>4</BOSS_ID>
  </Employee>
  <Employee>
    <EMPLOYEE_ID>4</EMPLOYEE_ID>
    <BOSS_ID>0</BOSS_ID>
  </Employee>
</ns0:Employees>

И вот пример того, как это должно выглядеть (на основе вышеуказанного источника):

<SyncCXPERSON>
  <CXPERSONSet>
    <PERSON>
        <PERSONID>1</PERSONID>
        <SUPERVISOR>2</SUPERVISOR>
        <BOSS>false</BOSS>
    </PERSON>
    <PERSON>
        <PERSONID>2</PERSONID>
        <SUPERVISOR>3</SUPERVISOR>
        <BOSS>true</BOSS>
    </PERSON>
    <PERSON>
        <PERSONID>3</PERSONID>
        <SUPERVISOR>4</SUPERVISOR>
        <BOSS>true</BOSS>
    </PERSON>
    <PERSON>
        <PERSONID>4</PERSONID>
        <SUPERVISOR></SUPERVISOR>
        <BOSS>true</BOSS>
    </PERSON>
  </CXPERSONSet>
</SyncCXPERSON>

Как вы видите, я должен проверить весь список повторенийзапишите 'Employee', чтобы проверить, является ли Employee также кем-то боссом, и в этом случае я установил в поле BOSS значение true.Это делается ниже XSLT:

<xsl:template match="Employees">
<SyncCXPERSON>
  <CXPERSONSet>
    <xsl:apply-templates select="Employee" />
  </CXPERSONSet>
</SyncCXPERSON>
</xsl:template>

<xsl:template match="Employee">
 <xsl:variable name="CURRENT_EMPLOYEE_ID" select="EMPLOYEE_ID" />
  <PERSON>
    <PERSONID>
      <xsl:value-of select="$CURRENT_EMPLOYEE_ID"/>
    </PERSONID>
    <SUPERVISOR>
      <xsl:value-of select="BOSS_ID"/>
    </SUPERVISOR>
    <BOSS>
      <xsl:choose>
        <xsl:when test="/Employees/Employee/BOSS_ID = $CURRENT_EMPLOYEE_ID">
          <xsl:value-of select="true"/>
        </xsl:when>
        <xsl:otherwise>
          <xsl:value-of select="false"/>
        </xsl:otherwise>
      </xsl:choose>
    </BOSS>
  </PERSON>
</xsl:template>

Это работает.Проблема в том, что я обрабатываю около 35600 сообщений (размер сообщения 3 МБ), и BizTalk требуется около 2 часов для обработки этого XSLT.

Итак, мой вопрос: я делаю это неправильно?Есть ли более эффективный способ сделать это?

1 Ответ

1 голос
/ 07 мая 2019

Попробуйте использовать клавишу , например:

XSLT 1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:key name="emp-by-boss" match="Employee" use="BOSS_ID" />

<xsl:template match="/*">
    <SyncCXPERSON>
        <CXPERSONSet>
            <xsl:apply-templates select="Employee" />
        </CXPERSONSet>
    </SyncCXPERSON>
</xsl:template>

<xsl:template match="Employee">
    <PERSON>
        <PERSONID>
            <xsl:value-of select="EMPLOYEE_ID"/>
        </PERSONID>
        <SUPERVISOR>
            <xsl:value-of select="BOSS_ID"/>
        </SUPERVISOR>
        <BOSS>
            <xsl:value-of select="boolean(key('emp-by-boss', EMPLOYEE_ID))"/>
        </BOSS>
    </PERSON>
</xsl:template>

</xsl:stylesheet>
...