首页 > 解决方案 > 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 的顶部,但没有成功。

有没有办法可以在一个映射中复制所有内容?

谢谢!问候,尼克

标签: xmlxsltxslt-1.0

解决方案


这个 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>来确定新元素名称。


推荐阅读