0

好的,我正在编辑原始帖子。我想我通过替换原始 XML 的名称标签做了很多尝试。无论如何,这是原始文件的摘录:

<EMPLOYEE_LIST>
   <EMPLOYEES>
      <EMPLOYEE>
         <EMPID>650000</EMPID>
         <FIRST_NAME>KEITH</FIRST_NAME>
         <MIDDLE_NAME>HUTCHINSON</MIDDLE_NAME>
         <LAST_NAME>ROGERS</LAST_NAME>
         <EMP_TYPE></EMP_TYPE>
         <EMP_REF_ID>500000</EMP_REF_ID>
         <JOINED_ON>2001-10-06</JOINED_ON>
         <COMMENTS>Miscellanous Comments</COMMENTS>
         <NATIONALITY>
            <VALUE>American</VALUE>
         </NATIONALITY>
         <EMP_AKA>
            <AKA_NAME>Danny</AKA_NAME>
         </EMP_AKA>
         <EMP_AKA>
            <AKA_NAME>Dan</AKANAME>
         </EMP_AKA>
         <EMP_AKA>
            <AKA_NAME>Ray</AKA_NAME>
         </EMP_AKA>
         <EMP_ADDR>
            <STREET> </STREET>
            <CITY> </CITY>
            <STATE> </STATE>
            <ZIP> </ZIP>
            <COUNTRY> </COUNTRY>
       </EMPLOYEE>
    </EMPLOYEES>
</EMPLOYEE_LIST>

我在使用上述 XML 时遇到的问题是,我找不到在一个属性下容纳多个 AKA(也称为)属性的方法,而我用于此转换的 XSL 如下:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
    <xsl:output method="xml" encoding="UTF-8" indent="yes"/>
    <xsl:strip-space elements="*"/>
        <xsl:template match="/EMPLOYEE_LIST">
            <employees>
                <xsl:apply-templates select="EMPLOYEES/node()"/>
            </employees>        
        </xsl:template>

        <xsl:template match="EMPLOYEE">
        <employee>
            <xsl:apply-templates select="*"/>
        </employee>
        </xsl:template>

        xsl:template match="EMPLOYEE/EMPID">
        <emp_id>
            <xsl:value-of select="."/>
        </emp_id>
        </xsl:template>

            <xsl:template match="EMPLOYEE/FIRST_NAME">
        <f_name>
            <xsl:value-of select="."/>
        </f_name>
        </xsl:template>

        <xsl:template match="EMPLOYEE/MIDDLE_NAME">
            <m_name>
                <xsl:value-of select="."/>
            </m_name>
        </xsl:template>

        <xsl:template match="EMPLOYEE/LAST_NAME">
            <l_name>
                <xsl:value-of select="."/>
            </l_name>
        </xsl:template>
        .
        .
        .
        .
        .
        <xsl:template match="EMPLOYEE/EMP_AKA">
        <aka_list>
            <xsl:for-each select="AKA_NAME">
                <aka>
                    <xsl:for-each select=".">
                        <xsl:apply-templates/>
                    </xsl:for-each>
                </aka>
            </xsl:for-each>
        </aka_list>
     </xsl:template>
</xsl:stylesheet>

当应用于我的 XML 时,上面给出的 XSL 给出了以下输出:

<?xml version="1.0" encoding="UTF-8"?>
<employees>
      <employee>
         <emp_id>111345</emp_id>
         <f_name>KEITH</f_name>
         <m_name>HUTCHINSON</m_name>
         <l_name>ROGERS</l_name>
         <aka_list>
            <aka>Danny</aka>
         </aka_list>
         <aka_list>
            <aka>Dan</aka>
         </aka_list>
         <aka_list>
            <aka>Ray</aka>
         </aka_list>
      </employee>
</employees>

这不是我想要实现的目标,因为我需要以下格式的数据:

<?xml version="1.0" encoding="UTF-8"?>
<employees>
      <employee>
         <emp_id>111345</emp_id>
         <f_name>KEITH</f_name>
         <m_name>HUTCHINSON</m_name>
         <l_name>ROGERS</l_name>
         <aka_list>
            <aka>Danny</aka>
            <aka>Dan</aka>
            <aka>Ray</aka>
         </aka_list>
      </employee>
</employees

有什么办法可以做到这一点?

展望未来,XML 中有大量元素,例如 AKA_NAME。

         <aka_list>
            <aka>Danny</aka>
         </aka_list>
         <aka_list>
            <aka>Dan</aka>
         </aka_list>
         <aka_list>
            <aka>Ray</aka>
         </aka_list>
         <aka_list>
            <aka>Danny_2</aka>
         </aka_list>
         <aka_list>
            <aka>Dan_2</aka>
         </aka_list>
         <aka_list>
            <aka>Ray_2</aka>
         </aka_list>

转换应该只继承前 5 个,第 6 个应该被截断,如下所示:

         <aka_list>
            <aka>Danny</aka>
            <aka>Dan</aka>
            <aka>Ray</aka>
            <aka>Danny_2</aka>
            <aka>Dan_2</aka>
         </aka_list>
4

2 回答 2

2

更新:完整 XML 的解决方案。

我尝试使用 XSLT 模板中的注释来解释解决方案。

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

    <!-- Ignore text nodes by default -->
    <xsl:template match="text()" />

    <!-- Transform tag name EMPLOYEES to employees -->
    <xsl:template match="EMPLOYEES">
        <employees>
            <xsl:apply-templates select="*" />
        </employees>
    </xsl:template>

    <!-- Transform tag name EMPLOYEE to employee -->
    <xsl:template match="EMPLOYEE">
        <employee>
            <xsl:apply-templates select="*" />
        </employee>
    </xsl:template>

    <!-- Transform tag name EMPID to emp_id -->
    <xsl:template match="EMPID">
        <emp_id>
            <xsl:value-of select="." />
        </emp_id>
    </xsl:template>

    <!-- Transform tag name FIRST_NAME to f_name -->
    <xsl:template match="FIRST_NAME">
        <f_name>
            <xsl:value-of select="." />
        </f_name>
    </xsl:template>

    <!-- Transform tag name MIDDLE_NAME to m_name -->
    <xsl:template match="MIDDLE_NAME">
        <m_name>
            <xsl:value-of select="." />
        </m_name>
    </xsl:template>

    <!-- Transform tag name LAST_NAME to l_name -->
    <xsl:template match="LAST_NAME">
        <l_name>
            <xsl:value-of select="." />
        </l_name>
    </xsl:template>

    <!-- When encounter the first EMP_AKA element, print itself and its following
         siblings with the same name within an aka_list element -->
    <xsl:template match="EMP_AKA[1]">
        <aka_list>
            <xsl:apply-templates select="AKA_NAME|following-sibling::EMP_AKA/AKA_NAME" mode="print"/>
        </aka_list>
    </xsl:template>

    <!-- Transform tag name EMP_AKA to aka -->
    <xsl:template match="AKA_NAME" mode="print">
        <aka>
            <xsl:value-of select="." />
        </aka>  
    </xsl:template>

</xsl:stylesheet>

更新 2:如果您不想使用模板模式,因为 AKA_NAME 将在其他地方匹配并以相同的方式处理,您可以将最后两个模板替换为以下模板:

<!-- When encounter the first EMP_AKA element, print itself and its following
     siblings with the same name within an aka_list element -->
<xsl:template match="EMP_AKA[1]">
    <aka_list>
        <xsl:apply-templates select="AKA_NAME|following-sibling::EMP_AKA/AKA_NAME" />
    </aka_list>
</xsl:template>

<!-- Exclude all EMP_AKA elements (except the first one because
     the previous template has higher precedence than this one) -->
<xsl:template match="EMP_AKA" />

<!-- Transform tag name EMP_AKA to aka -->
<xsl:template match="AKA_NAME">
    <aka>
        <xsl:value-of select="." />
    </aka>  
</xsl:template>

<xsl:template match="EMP_AKA" />

此代码产生与前一个相同的输出。


更新 3:OP 询问如何限制输出的 AKA_NAME 元素的数量。这是基于 UPDATE 2 的解决方案。只需将 UPDATE 2 中的最后一个模板替换为

<xsl:template match="EMP_AKA[1]">
    <aka_list>
        <xsl:apply-templates select="AKA_NAME|following-sibling::EMP_AKA[position() &lt; 5]/AKA_NAME" />
    </aka_list>
</xsl:template>

<!-- Transform tag name EMP_AKA to aka -->
<xsl:template match="AKA_NAME">
    <aka>
        <xsl:value-of select="." />
    </aka>  
</xsl:template>

<xsl:template match="EMP_AKA" />

原始答案

OP 编辑​​了帖子并完全更改了 XML 文件。以下是我之前的答案(删除它似乎不正确)。

如果您尝试将所有 <AKA> 元素组合成一个 <AKA_LIST> 元素(不清楚,因为您发布的输出与转换不匹配),那么您可以使用以下转换:

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

    <!-- Just for demo -->
    <xsl:template match="text()" />

    <!-- Match PERSON: create the list -->
    <xsl:template match="PERSON">
        <AKA_LIST>
            <xsl:apply-templates select="NAME/AKA" />
        </AKA_LIST>
    </xsl:template> 

    <!-- Outputs the AKA element, changing the tag name -->
    <xsl:template match="AKA">
        <aka>
            <xsl:value-of select="." />
        </aka>
    </xsl:template>

</xsl:stylesheet>

您将 XML 源转换为:

<AKA_LIST>
   <aka>ROSE PETAL</aka>
   <aka>JOHN FILTER</aka>
</AKA_LIST>
于 2013-02-26T11:14:26.957 回答
1

您的 XSLT 代码实际上令人困惑,因为您有一些标签,例如<AKA_LIST>. 所以我要输入和输出 XML 示例:这里是 XSLT 代码

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <xsl:output method="xml" indent="yes"/>

  <xsl:template match="@*|node()">
    <xsl:copy>
      <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
  </xsl:template>

  <xsl:template match="PERSON">
    <xsl:copy>
      <xsl:apply-templates select="NAME[1]"/>
    </xsl:copy>
  </xsl:template>

  <xsl:template match="NAME">
    <xsl:copy>
      <xsl:apply-templates select="/PEOPLE/PERSON/NAME/AKA"/>
    </xsl:copy>
  </xsl:template>


</xsl:stylesheet>

输入 XML:

<?xml version="1.0" encoding="utf-8"?>
<PEOPLE>
  <PERSON>
    <NAME>
      <REFERENCE>GOOD</REFERENCE>
      <AKA>ROSE PETAL</AKA>
      </NAME>
    <NAME>
      <REFERENCE>GOOD</REFERENCE>
      <AKA>JOHN FILTER</AKA>
      </NAME>
  </PERSON>
</PEOPLE>

输出:

<?xml version="1.0" encoding="utf-8"?>
<PEOPLE>
  <PERSON>
    <NAME>
      <AKA>ROSE PETAL</AKA>
      <AKA>JOHN FILTER</AKA>
    </NAME>
  </PERSON>
</PEOPLE>

解释:

<xsl:template match="@*|node()"> ......

上面的代码将标签从输入复制到输出AS IS *,其他模板匹配覆盖此..

<xsl:template match="PERSON"> ......

上面的代码只将一个<NAME>标签(第一个)复制到输出中。

<xsl:template match="NAME">
    <xsl:copy>
      <xsl:apply-templates select="/PEOPLE/PERSON/NAME/AKA"/>
      ......

上面的代码复制 <AKA> <NAME>. 由于我们只复制一个<NAME>标签,所有<AKA>标签都出现在<NAME>

于 2013-02-26T12:17:08.643 回答