3

我有一个这样的 XML 结构:

<root>
    <element>
        <name>Foo</name>
        <subelement>
             <key>1.1</key>
             <value>Lorem ipsum</value>
        </subelement>
        <subelement>
             <key>1.2</key>
             <value>Lorem ipsum dolor</value>
        </subelement>
    </element>
    <element>
        <name>Bar</name>
        <subelement>
             <key>7.3.4</key>
             <value>Seven three four</value>
        </subelement>
        <subelement>
             <key>7.3.8</key>
             <value>Seven three eight</value>
        </subelement>
        <subelement>
             <key>7.1</key>
             <value>Seven one</value>
        </subelement>
    </element>
</root>

我试图实现的是删除<subelement>除具有“最高”键的 s 之外的所有 s 。我似乎找不到任何方法来比较<key>s 内的<subelements>s <element>

生成的 XML 如下所示:

<root>
    <element>
        <name>Foo</name>
        <subelement>
             <key>1.2</key>
             <value>Lorem ipsum dolor</value>
        </subelement>
    </element>
    <element>
        <name>Bar</name>
        <subelement>
             <key>7.3.8</key>
             <value>Seven three eight</value>
        </subelement>
    </element>
</root>

任何提示都非常受欢迎。

4

2 回答 2

2

此 XSLT 2.0 转换适用于键中任意数量的“关键组件”以及任何关键组件的任何可能的正整数值

<xsl:stylesheet version="2.0"
 xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
 xmlns:xs="http://www.w3.org/2001/XMLSchema"
 xmlns:my="my:my">
 <xsl:output omit-xml-declaration="yes" indent="yes"/>
 <xsl:strip-space elements="*"/>

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

 <xsl:template match=
 "subelement
   [not(key eq my:max(../subelement/key))
   or
    key = preceding-sibling::subelement/key
   ]"/>

 <xsl:function name="my:max" as="xs:string">
   <xsl:param name="pValues" as="xs:string+"/>

   <xsl:sequence select=
    "if(not(distinct-values($pValues)[2]))
       then $pValues[1]
     else
      for $vMax1 in
             max(for $s in $pValues
                  return
                    xs:integer(substring-before(concat($s,'.'),'.'))
                 ),

          $vcntMax1Values in
             count($pValues[starts-with(., string($vMax1))])

        return
          if($vcntMax1Values eq 1)
            then $pValues[starts-with(., string($vMax1))]
                          [1]
            else
             for $submax in
                     (my:max(for $val in
                                    $pValues[starts-with(., string($vMax1))]
                                                [contains(., '.')],
                                    $subval in substring-after($val, '.')
                                return
                                    $subval
                             )
                      )
                 return
                   concat($vMax1, '.', $submax)

    "/>
 </xsl:function>
</xsl:stylesheet>

当此转换应用于以下 XML 文档(提供的文档,已扩展以使其更有趣)时:

<root>
    <element>
        <name>Foo</name>
        <subelement>
            <key>1.1</key>
            <value>Lorem ipsum</value>
        </subelement>
        <subelement>
            <key>1.2</key>
            <value>Lorem ipsum dolor</value>
        </subelement>
    </element>
    <element>
        <name>Bar</name>
        <subelement>
            <key>7.3.4</key>
            <value>Seven three four</value>
        </subelement>
        <subelement>
            <key>7.3.8</key>
            <value>Seven three eight</value>
        </subelement>
        <subelement>
            <key>7.3.8.1</key>
            <value>Seven three eight one</value>
        </subelement>
        <subelement>
            <key>7.3.8.1</key>
            <value>Seven three eight one</value>
        </subelement>
        <subelement>
            <key>7.1</key>
            <value>Seven one</value>
        </subelement>
        <subelement>
            <key>10.1</key>
            <value>Ten one</value>
        </subelement>
        <subelement>
            <key>10.1</key>
            <value>Ten one</value>
        </subelement>
    </element>
</root>

产生了想要的正确结果

<root>
   <element>
      <name>Foo</name>
      <subelement>
         <key>1.2</key>
         <value>Lorem ipsum dolor</value>
      </subelement>
   </element>
   <element>
      <name>Bar</name>
      <subelement>
         <key>10.1</key>
         <value>Ten one</value>
      </subelement>
   </element>
</root>

说明

这个通用解决方案的核心是一个函数my:max(),它给出一个非空的“结构化值”序列(结构化值是一个正整数序列,与“。”字符连接),产生一个(许多可能的)最大值。

这个函数是递归的。它执行以下操作

  1. 如果所有传递的值都相同,则返回其中的第一个。

  2. 否则,找到第一个组件的最大值。

  3. 如果只有一个值具有找到最大值的第一个分量,则返回此值。

  4. 否则,找到具有最大第一个分量的所有值的“尾部”的最大值(递归)。

  5. 最后,将在上一步中找到的最大的第一个分量与最大的“尾部”连接在一起——并返回这个值。

于 2012-04-05T03:29:04.753 回答
-1

以下版本 2.0 XSLT

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="2.0" 
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:template match="@* | node()">
        <xsl:copy>
            <xsl:apply-templates select="@* | node()"/>
        </xsl:copy>
    </xsl:template>

    <xsl:template match="element">
        <xsl:variable name="sorted">
            <xsl:for-each select="subelement">
                <xsl:sort select="key"/>
                <xsl:copy-of select="."/>
            </xsl:for-each>
        </xsl:variable>
        <xsl:variable name="highest" select="$sorted/subelement[count($sorted/subelement)]/key"/>
        <element>
            <xsl:copy-of select="name"/>
            <xsl:copy-of select="subelement[key=$highest]"/>
        </element>
    </xsl:template>

</xsl:stylesheet>

当应用于示例输入 XML 时,会生成所需的输出 XML

<?xml version="1.0" encoding="UTF-8"?>
<root>
    <element>
        <name>Foo</name>
        <subelement>
            <key>1.2</key>
            <value>Lorem ipsum dolor</value>
        </subelement>
    </element>
    <element>
        <name>Bar</name>
        <subelement>
            <key>7.3.8</key>
            <value>Seven three eight</value>
        </subelement>
    </element>
</root>
于 2012-04-05T00:33:30.233 回答