xml - XSLT 1.0:同时复制结构和子结构
问题描述
感谢您的时间。我会尽量保持简短和重点。
我的目标是使用XSLT 1.0来转换 XML 文件,以便它复制特定的节点。问题是,我需要复制也需要复制的结构的子结构。我希望下面的例子能让我的问题更清楚。
这里是源 XML:
<?xml version="1.0" encoding="UTF-8"?>
<parent_node>
<derived_from_a>
<element1>test</element1>
<element2>test</element2>
</derived_from_a>
<derived_from_b>
<element1>test</element1>
<element2>test</element2>
<derived_from_c>
<element3>test</element3>
<element4>test</element4>
</derived_from_c>
</derived_from_b>
</parent_node>
编辑:我需要将所有<derived_from_a>
节点复制到一个名为 的新节点中<a>
。节点<derived_from_b>
和<derived_from_c>
新节点<b>
和<c>
.
这是所需的输出:
<?xml version="1.0" encoding="UTF-8"?>
<parent_node>
<derived_from_a>
<element1>test</element1>
<element2>test</element2>
</derived_from_a>
<a>
<element1>test</element1>
<element2>test</element2>
</a>
<derived_from_b>
<element1>test</element1>
<element2>test</element2>
<derived_from_c>
<element3>test</element3>
<element4>test</element4>
</derived_from_c>
<c>
<element3>test</element3>
<element4>test</element4>
</c>
</derived_from_b>
<b>
<element1>test</element1>
<element2>test</element2>
<derived_from_c>
<element3>test</element3>
<element4>test</element4>
</derived_from_c>
<c>
<element3>test</element3>
<element4>test</element4>
</c>
</b>
</parent_node>
请注意,我将节点复制<derived_from_c>
到<c>
两个父结构(<derived_from_b>
和<b>
)中。这就是我需要的。
我编写了以下 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="//derived_from_a">
<!-- first copy structure with original name -->
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
<!-- then duplicate it -->
<a>
<xsl:copy-of select="child::node()"/>
</a>
</xsl:template>
<xsl:template match="//derived_from_b">
<!-- first copy structure with original name -->
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
<!-- then duplicate it -->
<b>
<xsl:copy-of select="child::node()"/>
</b>
</xsl:template>
<xsl:template match="//derived_from_c">
<!-- first copy structure with original name -->
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
<!-- then duplicate it -->
<c>
<xsl:copy-of select="child::node()"/>
</c>
</xsl:template>
</xsl:stylesheet>
这会生成以下输出:
<?xml version="1.0" encoding="UTF-8"?>
<parent_node>
<derived_from_a>
<element1>test</element1>
<element2>test</element2>
</derived_from_a>
<a>
<element1>test</element1>
<element2>test</element2>
</a>
<derived_from_b>
<element1>test</element1>
<element2>test</element2>
<derived_from_c>
<element3>test</element3>
<element4>test</element4>
</derived_from_c>
<c>
<element3>test</element3>
<element4>test</element4>
</c>
</derived_from_b>
<b>
<element1>test</element1>
<element2>test</element2>
<derived_from_c>
<element3>test</element3>
<element4>test</element4>
</derived_from_c>
</b>
</parent_node>
注意节点<c>
只存在于节点中<derived_from_b>
。不在节点中<b>
。我尝试将<derived_from_c>
转换移动到我的 XSLT 的顶部,但没有成功。
有没有办法可以在一个映射中复制所有内容?
谢谢!问候,尼克
解决方案
这个 XSLT 1.0 转换提供了您正在寻找的结果:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:strip-space elements="*" />
<xsl:output method="xml" indent="yes" />
<xsl:template match="node() | @*" name="identity">
<xsl:copy>
<xsl:apply-templates select="node() | @*" />
</xsl:copy>
</xsl:template>
<xsl:template match="*[starts-with(name(), 'derived_from_')]">
<xsl:call-template name="identity" />
<xsl:element name="{substring-after(name(), 'derived_from_')}">
<xsl:apply-templates select="node() | @*" />
</xsl:element>
</xsl:template>
</xsl:stylesheet>
即
<parent_node>
<derived_from_a>
<element1>test</element1>
<element2>test</element2>
</derived_from_a>
<a>
<element1>test</element1>
<element2>test</element2>
</a>
<derived_from_b>
<element1>test</element1>
<element2>test</element2>
<derived_from_c>
<element3>test</element3>
<element4>test</element4>
</derived_from_c>
</derived_from_b>
<b>
<element1>test</element1>
<element2>test</element2>
<derived_from_c>
<element3>test</element3>
<element4>test</element4>
</derived_from_c>
<c>
<element3>test</element3>
<element4>test</element4>
</c>
</b>
</parent_node>
这里的诀窍是同时提供身份模板 amatch
和 a name
。这样,当 XSLT 处理器遍历您的输入文档时,它既可以“拉式”调用,也可以手动通过<xsl:call-template>
.
手动调用使我们能够处理当前节点的后代,而不仅仅是复制它们,这满足了您“复制也需要复制的结构的子结构”的要求。
如果您的输出元素名称 ( <a>
) 实际上不是输入元素名称 ( <derived_from_a>
) 的子字符串,您可能会想复制工作模板。
为了避免这种重复,您可以将映射嵌入到自定义元素中的 XSLT 中,并使用自引用通过document('')
动态提取新名称:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:my="http://tempuri.org"
>
<my:map>
<name from="derived_from_a" to="x" />
<name from="derived_from_b" to="y" />
<name from="derived_from_c" to="z" />
</my:map>
<!-- ... -->
<xsl:template match="derived_from_a|derived_from_b|derived_from_c">
<!-- ... -->
<xsl:element name="{document('')/*/my:map/name[@from = name(current())]/@to}">
<!-- ... -->
</xsl:element>
</xsl:template>
</xsl:stylesheet>
使用自定义元素添加配置有时会非常有用。另一种选择是通过<xsl:param>
来自调用代码的映射传递映射,具体取决于您使用的 XSLT 工具的功能。第三个选项是使用<xsl:variable>
包含 an<xsl:choose>
来确定新元素名称。