如何在保持结构的同时使用 xslt 重新排序 xml 文件

Posted

技术标签:

【中文标题】如何在保持结构的同时使用 xslt 重新排序 xml 文件【英文标题】:How to re-order a xml-file with xslt while keeping structure 【发布时间】:2020-06-03 13:26:47 【问题描述】:

我知道这是一个基本问题,但我对 XSL 很陌生,而且我无法让它工作。我有一个具有给定结构的 XML 文件,我想在保持结构的同时基于一个节点按字母顺序重新排列 XML。在 Adob​​e InDesign 中导入 XML 时,我需要 XSLT 样式表来重新排序它。

我的文件:

<?xml version="1.0" encoding="UTF-8"?>
<Root>
    <Employee>
        <First_Name>Andrew</First_Name>
        <Last_Name>Miller</Last_Name>
        <Salary>100000</Salary>
        <Performance>8</Performance>
    </Employee>
    <Employee>
        <First_Name>Betsy</First_Name>
        <Last_Name>Clarke</Last_Name>
        <Salary>105000</Salary>
        <Performance>10</Performance>
    </Employee>
    <Employee>
        <First_Name>Donald</First_Name>
        <Last_Name>Abernathey</Last_Name>
        <Salary>95000</Salary>
        <Performance>7</Performance>
    </Employee>
</Root>

我想按照标签&lt;Last_name&gt; 的字母顺序重新排列文档。我想要的输出是:

<?xml version="1.0" encoding="UTF-8"?>
<Root>
    <Employee>
        <First_Name>Donald</First_Name>
        <Last_Name>Abernathey</Last_Name>
        <Salary>95000</Salary>
        <Performance>7</Performance>
    </Employee>
    <Employee>
        <First_Name>Betsy</First_Name>
        <Last_Name>Clarke</Last_Name>
        <Salary>105000</Salary>
        <Performance>10</Performance>
    </Employee>
    <Employee>
        <First_Name>Andrew</First_Name>
        <Last_Name>Miller</Last_Name>
        <Salary>100000</Salary>
        <Performance>8</Performance>
    </Employee>
</Root>

我找到了一些解决方案来重新排列我的输出,但它总是针对输出 html 而不是 XML。

更新:按照这里的要求,我试图解决这个问题。我卡住了,因为它现在将所有内容都复制到一个标签 &lt;Employee&gt;

<?xml version="1.0" encoding="UTF-8"?>
<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:template match="/">
  <Root>
  <Employee>
      <xsl:for-each select="Root/Employee">
      <xsl:sort select="Last_Name"/>
        <First_Name><xsl:value-of select="First_Name"/></First_Name>
        <Last_Name><xsl:value-of select="Last_Name"/></Last_Name>
        <Salary><xsl:value-of select="Salary"/></Salary>
        <Performance><xsl:value-of select="Performance"/></Performance>
      </xsl:for-each>
  </Employee>
  </Root>
</xsl:template>

</xsl:stylesheet>

【问题讨论】:

在处理Employee 元素的地方,您想使用xsl:sort select="LastName"。无论您只是使用xsl:apply-templatesxsl:for-eachxsl:perform-sort。所以试一试,告诉我们你在哪里卡住了,结果是 XML 还是 HTML 并不重要。 除非您向我们展示您的最佳尝试,否则我们看不到您遇到的困难。我们可以为您编写代码,但这并不能教给您任何东西。当您说“我找到了一些生成 HTML 的解决方案”时,这表明您正在尝试找到完全符合您要求的代码示例,而不是尝试学习可以结合起来为任意问题创建解决方案的基本概念.你可能需要做更多的阅读。 @MichaelKay 你说的太对了。我必须做更多的阅读。你有什么好的资源可以指导我吗? 每个人的学习方式都不一样。如果我必须学习一门新语言,我总是先给自己买一本书,然后从头到尾略读一遍,以了解其中的内容。有很多关于 XSLT 的好书——包括我自己的。 感谢@MichaelKay 的提示。 【参考方案1】:

或者简单地说:

<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:template match="/Root">
    <xsl:copy>
        <xsl:for-each select="Employee">
            <xsl:sort select="Last_Name"/>
            <xsl:copy-of select="."/>
        </xsl:for-each>
    </xsl:copy>
</xsl:template>

</xsl:stylesheet>

【讨论】:

【参考方案2】:

如果您将 XML 转换为想要保留部分甚至大部分内容的 XML,我建议从身份转换模板开始,然后添加执行您想要进行的更改所需的模板,例如在你的情况下排序:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    version="1.0">

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

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

  <xsl:template match="Root">
    <xsl:copy>
        <xsl:apply-templates>
            <xsl:sort select="Last_Name"/>
        </xsl:apply-templates>
    </xsl:copy>
  </xsl:template>

</xsl:stylesheet>

https://xsltfiddle.liberty-development.net/6rexjig

【讨论】:

【参考方案3】:

真的接近解决方案:

<?xml version="1.0" encoding="UTF-8"?>
<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:template match="/">
  <Root>
      <xsl:for-each select="Root/Employee">

      <xsl:sort select="Last_Name"/>
      <Employee>
        <First_Name><xsl:value-of select="First_Name"/></First_Name>
        <Last_Name><xsl:value-of select="Last_Name"/></Last_Name>
        <Salary><xsl:value-of select="Salary"/></Salary>
        <Performance><xsl:value-of select="Performance"/></Performance>
       </Employee>
      </xsl:for-each>
  </Root>
</xsl:template>

</xsl:stylesheet>

【讨论】:

哦,伙计。只需将 移动到 for-each 中即可解决问题...

以上是关于如何在保持结构的同时使用 xslt 重新排序 xml 文件的主要内容,如果未能解决你的问题,请参考以下文章

XSLT 循环排序将以下节点向上移动

如何使用弹性框重新排序 div?

如何获取具有相同名称的元素并根据 XSLT 中的子节点值对它们进行排序

在遵循特定结构的同时按列对多索引进行排序

如何对适配器中的输入请求执行 xslt 转换

XSLT - 创建一个应该包含唯一和排序数据的变量