同时复制结构和子结构

问题描述 投票:0回答:1

我的目标是使用 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 的顶部,但没有成功。

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

xml xslt xslt-1.0
1个回答
2
投票

此 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>

这里的技巧是为身份模板提供

match
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>
<xsl:choose>
来确定新元素名称。

© www.soinside.com 2019 - 2024. All rights reserved.