使用XSLT 1.0合并并覆盖重复的xml节点

时间:2016-06-03 09:58:48

标签: xslt

我有源XML

<Records>
  <Data>
    <RecordType>New</RecordType>
    <Number>4734122946</Number>
    <DateOfBirth>20160506</DateOfBirth>
    <Title>Mr</Title>
    <ChangeTimeStamp>20160101010001</ChangeTimeStamp>
    <SerialChangeNumber>01</SerialChangeNumber>
  </Data>
  <Data>
    <RecordType>New</RecordType>
    <Number>4734122946</Number>
    <LastName>Potter</LastName>
    <DateOfBirth>20160506</DateOfBirth>
    <ChangeTimeStamp>20160101010002</ChangeTimeStamp>
    <SerialChangeNumber>01</SerialChangeNumber>
  </Data>
</Records>

我想使用XSLT 1.0生成以下输出

<Contact>
  <Number>4734122946</Number>
  <Title>Mr</Title>
  <LastName>Potter</LastName>
  <BirthDate>20160506</BirthDate>
  <ChangeTimeStamp>20160101010002</ChangeTimeStamp>
</Contact>

XSLT必须根据Number字段将Data记录的子节点分组并合并为一个。此外,如果存在相同的元素,则应使用ChangeTimeStamp元素来确定最新的更改并使用该元素。

我尝试了下面的XSLT。但我无法接近输出。

&#13;
&#13;
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:msxsl="urn:schemas-microsoft-com:xslt" exclude-result-prefixes="msxsl">
  <xsl:output method="xml" indent="yes" />

  <xsl:key name="groups" match="Data" use="Number"/>
  <xsl:key name="sortGroup" match="Data" use ="ChangeTimeStamp"/>

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

  <xsl:template match="Records">
    <xsl:for-each select="Data[generate-id() = generate-id(key('groups',Number))]">
      <Contact>
        <Number>
          <xsl:value-of select="Number"/>
        </Number>
        <xsl:for-each select="key('groups',Number)">
          <xsl:for-each select="key('sortGroup',ChangeTimeStamp)">
            <xsl:sort select="sortGroup" order="ascending"/>
            <xsl:if test="Title">
              <Title>
                <xsl:value-of select ="Title"/>
              </Title>
            </xsl:if>
            <xsl:if test="LastName">
              <LastName>
                <xsl:value-of select="LastName"/>
              </LastName>
            </xsl:if>
            <BirthDate>
              <xsl:value-of select="DateOfBirth"/>
            </BirthDate>
          </xsl:for-each>
        </xsl:for-each>
      </Contact>
    </xsl:for-each>
  </xsl:template>
</xsl:stylesheet>
&#13;
&#13;
&#13;

感谢您的帮助。

1 个答案:

答案 0 :(得分:0)

  

XSLT必须将数据记录的子节点分组并合并到   一个基于数字字段。如果有相同的元素   现在,它应该使用ChangeTimeStamp元素来弄清楚   最新的变化并使用该元素。

为此,我相信你会想做类似的事情:

<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:key name="group" match="Data" use="Number"/>
<xsl:key name="item" match="Data/*" use="concat(../Number, '|', name())"/>

<xsl:template match="/Records">
    <root>
        <xsl:for-each select="Data[generate-id() = generate-id(key('group', Number)[1])]">
            <Contact>
                <xsl:for-each select="key('group', Number)/*[generate-id() = generate-id(key('item', concat(../Number, '|', name()))[1])]">
                    <xsl:for-each select="key('item', concat(../Number, '|', name()))">
                        <xsl:sort select="../ChangeTimeStamp" data-type="number" order="descending"/>
                            <xsl:if test="position()=1">
                                <xsl:copy-of select="."/>
                            </xsl:if>
                    </xsl:for-each>
                </xsl:for-each>
            </Contact>
        </xsl:for-each>
    </root>
</xsl:template>

</xsl:stylesheet>

如果您只想包含某些数据项和/或希望它们按特定顺序显示,则必须进行一些调整。如果您有一个所有可能的数据项名称列表(例如Title,LastName,DateOfBirth等),那么这可能更简单。