XSL - 迭代元素并根据另一个xml文件

时间:2015-06-15 07:14:29

标签: xml xslt

我有一个XML文件,其中包含多个Shape元素,每个元素都包含一个包含Material属性的子Code元素。我想根据从单独的XML文件获取的值更新每个Code元素的Material属性。我遇到的问题是更新源文件也有多个元素。

这是主要的XML文件 - main.xml:

<?xml version="1.0"?>
<!--Top-->
<Top>
  <Shapes>
    <!--Shape-->
    <Shape Code="Penisola" Description="Rectangle">
      <!--Material-->
      <Material Code="MAT01000257" ></Material>
    </Shape>
    <!--Shape-->
    <Shape Code="Penisola" Description="Rectangle">
      <!--Material-->
      <Material Code="MAT01000258" ></Material>
    </Shape>
  </Shapes>
  <Texts></Texts>
</Top>

...和我的更新源文件 - updatesource.xml:

<?xml version="1.0"?>
<Job>
  <Job_Number>B90512</Job_Number>
  <Job_Address>2nd Floor/ 28-32 Albert Road   VIC 3205</Job_Address>
  <Benchtops>
    <Benchtop>
      <Material>Material 1</Material>
    </Benchtop>
    <Benchtop>
      <Material>Material 2</Material>
    </Benchtop>
  </Benchtops>
</Job>

期望的结果是 - output.xml:

<?xml version="1.0"?>
<!--Top-->
<Top>
  <Shapes>
    <!--Shape-->
    <Shape Code="Penisola" Description="Rectangle">
      <!--Material-->
      <Material Code="Material 1" ></Material>
    </Shape>
    <!--Shape-->
    <Shape Code="Penisola" Description="Rectangle">
      <!--Material-->
      <Material Code="Material 2" ></Material>
    </Shape>
  </Shapes>
  <Texts></Texts>
</Top>

我想我需要遍历主XML和更新源XML,但我不确定如何做到这一点。到目前为止,我的尝试仅使用更新源文件中第一个Code元素的值更新了Material个属性。

一些额外的信息; main.xml文件和updatesource.xml文件将分别具有匹配数量的ShapeBenchtop元素,但是每个元素的数量将随着我处理的每个作业而变化,因此解决方案将需要动态地处理存在的元素数量。

2 个答案:

答案 0 :(得分:1)

符合您要求的XSL转换可能如下所示:

<?xml version="1.0" encoding="utf-16" ?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
    <xsl:output method="xml"/>

    <!-- the update source file: -->
    <xsl:param name="usource" select="'updatesource.xml'"/>

    <xsl:template match="Material">
        <!-- Material's position: -->
        <xsl:variable name="pos">
            <xsl:value-of select="count(preceding::Material)+1"/>
        </xsl:variable>

        <xsl:copy>
            <xsl:apply-templates select="@*"/>

            <!-- update Code-attribute according to pos-th Material in $usource --> 
            <xsl:attribute name="Code">
                <xsl:value-of select="(document($usource)//Material)[position()=$pos]"/>
            </xsl:attribute>
            <xsl:apply-templates/>
        </xsl:copy>
    </xsl:template>

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

要点是检查源Material中的位置。然后从updatesoure.xml中选择相应的一个。

答案 1 :(得分:0)

怎么样:

XSLT 1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:param name="update-path" select="'updatesource.xml'" />

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

<xsl:template match="Shape">
    <xsl:variable name="i" select="position()" />
    <xsl:copy>
        <xsl:copy-of select="@*"/>
        <xsl:copy-of select="document($update-path)/Job/Benchtops/Benchtop[$i]/Material"/>
    </xsl:copy>
</xsl:template>

</xsl:stylesheet>
  

但是这不会更新Code属性,它会替换   现有的Material元素与新的元素。我需要保持   现有元素和其中的所有其他属性。

然后我建议你这样做:

<xsl:template match="Material/@Code">
    <xsl:variable name="i" select="count(../../preceding-sibling::Shape) + 1" />
    <xsl:attribute name="Code">
        <xsl:value-of select="document($update-path)/Job/Benchtops/Benchtop[$i]/Material"/>
    </xsl:attribute>
</xsl:template>

注意:我不是特别喜欢计算前面的兄弟姐妹而不是使用position()函数,但XSLT 1.0中的替代方案相当尴尬,所以除非你有大量的记录,你也可以忍受它。