好的,我正在编辑原帖。我想我通过替换原始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>
上面给出的XSL在应用于我的XML时给出了以下输出:
<?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>
转换应仅结转前五名,第六名应截断为:
<aka_list>
<aka>Danny</aka>
<aka>Dan</aka>
<aka>Ray</aka>
<aka>Danny_2</aka>
<aka>Dan_2</aka>
</aka_list>
答案 0 :(得分: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将在其他位置匹配并以相同方式处理,您可以将这两个模板替换为以下模板:< / p>
<!-- 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的解决方案。只需用UPRATE 2替换最后一个模板
<xsl:template match="EMP_AKA[1]">
<aka_list>
<xsl:apply-templates select="AKA_NAME|following-sibling::EMP_AKA[position() < 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文件。以下是我之前的回答(似乎没有将其删除)。
如果您尝试将所有&lt; AKA&gt;分组。将元素转换为&lt; AKA_LIST&gt; element(不清楚,因为您发布的输出与转换不匹配),那么您可以使用以下转换:
<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>
答案 1 :(得分: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>
TAG(第一个)复制到输出中..
<xsl:template match="NAME">
<xsl:copy>
<xsl:apply-templates select="/PEOPLE/PERSON/NAME/AKA"/>
......
以上代码复制 <AKA>
下的所有 <NAME>
标记。由于我们只复制了一个<NAME>
代码,因此所有<AKA>
代码都会显示在<NAME>
下