当前位置: 首页 > 知识库问答 >
问题:

通过XSLT复制具有不同属性的节点或根据条件将子节点添加到其他节点

轩辕煜
2023-03-14

我正在研究一个xslt,它将用于翻译一组xmls文件,一个示例输入xml类似于:-

<?xml version="1.0" encoding="UTF-8"?>  
<bookstore>

  <book id="A12">
    <bookattribute  name="abc"  price="200" />
    <bookattribute  name="xyz"  price="300" />
    <bookattribute  name="pqr"  price="400" />
    <bookattribute  name="lnz"  price="500" />
  </book>

  <book id="B12">
    <bookattribute  name="cpz"  price="300" />
    <bookattribute  name="rts"  price="800" />
  </book>


  <novel id="AA12">
    <bookattribute  name="yps"  price="200" />
    <bookattribute  name="udv"  price="600" />
  </novel>

  <!-- book node with id=AA12 may or may not be present in the xml  -->

  <book id="AA12">
    <bookattribute  name="abc"  price="200" />
    <bookattribute  name="aps"  price="600" /> 
  </book>

</bookstore>

我想通过创建属性为id=“A12”的节点“book”的副本对其进行转换,条件是,如果属性为id=AA12的节点book不存在,则创建节点“book”的副本(属性为id=“A12”),并在副本中将id更改为“AA12”(因此其属性值更改的节点副本),但如果属性为book的节点(id=“AA12”)已经存在于xml中,然后添加那些在“book”(attribute id=“AA12”)中不存在的book的子节点(id=A12),这里我只想添加那些在“book”中不存在的子节点(以bookattribute“name”作为键)book@id=AA12),例如,如果子节点bookattribute name=“abc”已存在于book@id=AA12则不应再次将其添加到其中,此外,它们可能是某些其他元素节点,如小说或电子书,也可能具有属性id=“AA12”。必须按原样复制这些节点。

因此,在这种情况下输出(上面的输入文件包含节点“book”(属性id=“AA12”))

<bookstore>
            <book id="A12">
                <bookattribute  name="abc"  price="200" />
                <bookattribute  name="xyz"  price="300" />
                <bookattribute  name="pqr"  price="400" />
                <bookattribute  name="lnz"  price="500" />
              </book>

              <book id="B12">
                <bookattribute  name="cpz"  price="300" />
                <bookattribute  name="rts"  price="800" />
              </book>

  <novel id="AA12">
    <bookattribute  name="yps"  price="200" />
    <bookattribute  name="udv"  price="600" />
  </novel>

               <book id="AA12">
                <bookattribute  name=aps  price=600 />
                <bookattribute  name="abc"  price="200" />
                <bookattribute  name="xyz"  price="300" />
                <bookattribute  name="pqr"  price="400" />
                <bookattribute  name="lnz"  price="500" />
              </book>
    </bookstore>

输出,以防(输入文件不包含节点“book”(属性id=“AA12”))

    <bookstore>

                  <book id="A12">
                    <bookattribute  name="abc"  price="200" />
                    <bookattribute  name="xyz"  price="300" />
                    <bookattribute  name="pqr"  price="400" />
                    <bookattribute  name="lnz"  price="500" />
                  </book>

                  <book id="B12">
                    <bookattribute  name="cpz"  price="300" />
                    <bookattribute  name="rts"  price="800" />
                  </book>

 <novel id="AA12">
        <bookattribute  name="yps"  price="200" />
        <bookattribute  name="udv"  price="600" />
      </novel>

                   <book id="AA12">
                    <bookattribute  name="abc"  price="200" />
                    <bookattribute  name="xyz"  price="300" />
                    <bookattribute  name="pqr"  price="400" />
                    <bookattribute  name="lnz"  price="500" />
                  </book>
        </bookstore>

我已经能够创建xslt,如下所示,它可以根据需要创建具有更改属性的副本,但我无法计算出添加子节点的方法,如果存在id为“AA12”的节点簿,任何指针都会有帮助

我的xslt:-

<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
                version="1.0">
                
<xsl:output method="xml"/>
    
    <xsl:template match="*">
        <xsl:copy>
            <xsl:copy-of select="@*"/>
            <xsl:apply-templates/>
       </xsl:copy>
    </xsl:template>

    <xsl:template match="comment()|processing-instruction()">
        <xsl:comment>
            <xsl:value-of select="."/>
        </xsl:comment>
    </xsl:template>
    
    <xsl:template match="book[@id='A12']">
        <xsl:copy>
            <xsl:attribute name="id">A12</xsl:attribute>
            <xsl:apply-templates />
        </xsl:copy>
        <xsl:copy>
            <xsl:attribute name="id">AA12</xsl:attribute>
            <xsl:apply-templates />
        </xsl:copy>
    </xsl:template>
    
</xsl:stylesheet>

共有1个答案

冀鸿才
2023-03-14

发布XML时请小心,您的输入文档格式不好。

使用键通过ID识别book元素。然后,原理如下:

  • 找到书店元素并将其复制到输出

XSLT样式表

编辑我已经编辑了样式表以响应

此xslt可能需要少量修复,因为它可能会复制子节点,例如,如果子节点已经存在于中book@id=“A12”和“AA12”,需要一个过滤器以避免复制重复的子节点

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

    <xsl:output method="xml" indent="yes"/>
    <xsl:strip-space elements="*"/>

    <xsl:key name="book-by-id" match="book" use="@id"/>

    <xsl:template match="bookstore">
        <xsl:copy>
          <xsl:apply-templates select="@*|node()[not(self::book and @id = 'AA12')]"/>
          <book id="AA12">
              <xsl:copy-of select="key('book-by-id','AA12')/*"/>
              <xsl:for-each select="key('book-by-id','A12')/*">
                  <xsl:if test="not(./@name = key('book-by-id','AA12')/*/@name)">
                      <xsl:copy-of select="."/>
                  </xsl:if>
              </xsl:for-each>
          </book>
      </xsl:copy>
    </xsl:template>

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

</xsl:transform>

替代解决方案:

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

    <xsl:output method="xml" indent="yes"/>
    <xsl:strip-space elements="*"/>

    <xsl:key name="book-by-id" match="book" use="@id"/>
    <xsl:key name="book-attribute-by-name" match="bookattribute" use="@name"/>

    <xsl:template match="bookstore">
        <xsl:copy>
          <xsl:apply-templates select="@*|node()[not(@id = 'AA12')]"/>
          <book id="AA12">
              <xsl:apply-templates select="key('book-by-id','AA12')/*,key('book-by-id','A12')/*"/>
          </book>
      </xsl:copy>
    </xsl:template>

    <xsl:template match="key('book-by-id','A12')/*[@name = key('book-by-id','AA12')/*/@name]"/>

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

</xsl:transform>

XML输出

如果已经有一本书籍元素带有id=“AA12”

<bookstore>
   <book id="A12">
      <bookattribute name="abc" price="200"/>
      <bookattribute name="xyz" price="300"/>
      <bookattribute name="pqr" price="400"/>
      <bookattribute name="lnz" price="500"/>
   </book>
   <book id="B12">
      <bookattribute name="cpz" price="300"/>
      <bookattribute name="rts" price="800"/>
   </book>
   <!-- book node with id=AA12 may or may not be present in the xml  -->
   <book id="AA12">
      <bookattribute name="abc" price="200"/>
      <bookattribute name="xyz" price="300"/>
      <bookattribute name="pqr" price="400"/>
      <bookattribute name="lnz" price="500"/>
      <bookattribute name="aps" price="600"/>
   </book>
</bookstore>

如果没有这种因素:

<bookstore>
   <book id="A12">
      <bookattribute name="abc" price="200"/>
      <bookattribute name="xyz" price="300"/>
      <bookattribute name="pqr" price="400"/>
      <bookattribute name="lnz" price="500"/>
   </book>
   <book id="B12">
      <bookattribute name="cpz" price="300"/>
      <bookattribute name="rts" price="800"/>
   </book>
   <!-- book node with id=AA12 may or may not be present in the xml  -->
   <book id="AA12">
      <bookattribute name="abc" price="200"/>
      <bookattribute name="xyz" price="300"/>
      <bookattribute name="pqr" price="400"/>
      <bookattribute name="lnz" price="500"/>
   </book>
</bookstore>
 类似资料:
  • 我已经创建了一个XSLT,我想知道如何在一组标记之间复制所有节点,并在底部添加另一个标记。我创建了XSLT,它具有确定要添加哪个标记以及应该调用什么的所有逻辑。然而,我现在遇到的问题是,我不能复制所有其他标签了。以下是有关文件: XSLT 输入 电流输出 期望的产出

  • 我有一个XML: 我想向根元素:/doc添加一个属性(名称空间),以便输出如下所示: 我尝试了三种xslt(跳过了默认的“复制所有”部分以减少问题的长度)。 xslt1:见下文,问题在于将空名称空间xmlns=”“添加到/doc的所有子节点(即:/doc/tag1和/doc/tag2) Xslt2:见下文,问题是“ns”被添加到根节点:和 xslt3:请参见下文,问题是报告了错误:未定义名称空间前

  • 我正在寻找下面输入和输出XML的XSLT(1.0)代码。 在输出XML中,C6元素下可以有任何子节点。在下面的XML中,我放了CN元素,但它可以是任何名称。 输入XML- 所需的输出XML- 先谢谢你。

  • 我想从父节点复制到子节点。我真的不确定这是如何实现的。 我的源xml 我想得到输出为 我想要XSLT1.0中的解决方案。 我想将这些节点复制到子节点 谢谢。

  • 我需要以下xslt代码方面的帮助。我的意见如下: 我的预期输出为: 我曾尝试使用*/text()获取值节点的值,但我只从第一个孩子获得文本。将来我有很多这样的子元素。 提前谢谢。 你好,Minakshi

  • 我需要将子元素复制到父元素中。 输入 期望输出 我尝试的内容(输出与输入保持相同): 我肯定会错过一些非常简单的事情。子元素与父元素具有相同的名称,这应该不是问题?