通过多个属性对XML文件进行排序

时间:2010-06-05 10:36:57

标签: xml xslt

我想通过多个属性(首先是T,然后是L)对“自由格式”XML文件进行排序。 XML有点复杂,结构如下所示:

<?xml version="1.0" encoding="utf-8"?>
<wb xmlns:cf="http://www.macromedia.com/2004/cfform" xmlns:a="urn:dummy">
  <a:form name="chart">
    <a:fieldset FIELD="a" FIELDNAME="FieldSet1">
      <a:select1 FIELDNUMBER="01" L="1" T="2" />
      <a:input FIELDNUMBER="02" INDEX="4" L="200" T="1" />
    </a:fieldset>
    <a:fieldset FIELD="b" FIELDNAME="FieldSet1">
      <a:select1 FIELDNUMBER="03" T="3" L="1" />
      <a:input FIELDNUMBER="04" INDEX="7" T="4" L="200" />
      <a:fieldset FIELD="c" FIELDNAME="FieldSet1">
        <a:input FIELDNUMBER="05" T="10" INDEX="6" L="400" />
        <a:input FIELDNUMBER="06" T="8" INDEX="8" L="200" />
      </a:fieldset>
    </a:fieldset>
    <a:input FIELDNUMBER="08" INDEX="3" L="3" T="5" />
    <a:input FIELDNUMBER="09" INDEX="2" L="2" T="5" />
  </a:form>
</wb>

PS:

  1. 根元素是wb,后面跟着一个:form
  2. L和T总是在名称空间a中有标记的元素中找到,唯一的例外是:没有L和T的字段集
  3. a:fieldset可以有多个名称空间的子项a,包括另一个a:fieldset
  4. 在对字段集中的子项进行排序时,这些子项需要保持附加到其当前父项。
  5. 结果输出应如下:

    <?xml version="1.0" encoding="utf-8"?>
    <wb xmlns:cf="http://www.macromedia.com/2004/cfform" xmlns:a="urn:dummy">
      <a:form name="chart">
        <a:fieldset FIELD="a" FIELDNAME="FieldSet1">
          <a:input FIELDNUMBER="02" INDEX="4" L="200" T="1" />
          <a:select1 FIELDNUMBER="01" L="1" T="2" />
        </a:fieldset>
        <a:fieldset FIELD="b" FIELDNAME="FieldSet1">
          <a:select1 FIELDNUMBER="03" T="3" L="1" />
          <a:input FIELDNUMBER="04" INDEX="7" T="4" L="200" />
          <a:fieldset FIELD="c" FIELDNAME="FieldSet1">
            <a:input FIELDNUMBER="06" T="8" INDEX="8" L="200" />
            <a:input FIELDNUMBER="05" T="10" INDEX="6" L="400" />
          </a:fieldset>
        </a:fieldset>
        <a:input FIELDNUMBER="09" INDEX="2" L="2" T="5" />
        <a:input FIELDNUMBER="08" INDEX="3" L="3" T="5" />
      </a:form>
    </wb>
    

    为了更好地理解,我们可以假设L表示左,T表示Top。所以,这个想法是,当我查看转换后的XML时,我可以立即注意到哪些元素在前面。

    你对此有何看法?

1 个答案:

答案 0 :(得分:0)

试试这个:(如果我没有正确理解你的问题,做一些改变)

<xsl:template match="/wb">
    <xsl:copy>
      <xsl:copy-of select="@*"/>
      <xsl:apply-templates mode="aaa" select="."/>
    </xsl:copy>
  </xsl:template>

  <xsl:template match="*" mode="aaa">
    <xsl:if test="child::*[1][not(@T) or not(@L)]">
      <xsl:for-each select="child::*[not(@T) or not(@L)]">
        <xsl:copy>
          <xsl:copy-of select="@*"/>
          <xsl:apply-templates mode="aaa" select="."/>
        </xsl:copy>
      </xsl:for-each>
    </xsl:if>
    <xsl:variable name="sorted1">
      <xsl:for-each select="a:*[@T and @L]">
        <xsl:sort select="@T" data-type="number" order="ascending"></xsl:sort>
        <xsl:copy>
          <xsl:copy-of select="@*"/>
        </xsl:copy>
      </xsl:for-each>
    </xsl:variable>
    <xsl:for-each select="msxsl:node-set($sorted1)/*[not(@T=preceding-sibling::*/@T)]">
      <xsl:variable name="tval" select="@T"/>
      <xsl:for-each select="msxsl:node-set($sorted1)/*[@T=$tval]">
        <xsl:sort select="@L" data-type="number" order="ascending"></xsl:sort>
        <xsl:copy>
          <xsl:copy-of select="@*"/>
        </xsl:copy>
      </xsl:for-each>
    </xsl:for-each>

    <xsl:if test="not(child::*[1][not(@T) or not(@L)])">
      <xsl:for-each select="child::*[not(@T) or not(@L)]">
        <xsl:copy>
          <xsl:copy-of select="@*"/>
          <xsl:apply-templates mode="aaa" select="."/>
        </xsl:copy>
      </xsl:for-each>
    </xsl:if>

  </xsl:template>