Warning: file_get_contents(/data/phpspider/zhask/data//catemap/0/xml/15.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Xml XSLT对两个不同节点的子节点进行排序_Xml_Xslt - Fatal编程技术网

Xml XSLT对两个不同节点的子节点进行排序

Xml XSLT对两个不同节点的子节点进行排序,xml,xslt,Xml,Xslt,您好,我是xslt新手,我遇到了一个问题,我需要对子节点进行排序,不管它们有什么父节点,并显示排列的内容。因此,XML文件是 <?xml version="1.0" encoding="ISO-8859-1" standalone="yes" ?> <report> <rowsbypage>5</rowsbypage><!--added to complete the solution--> <contenido&

您好,我是xslt新手,我遇到了一个问题,我需要对子节点进行排序,不管它们有什么父节点,并显示排列的内容。因此,XML文件是

<?xml version="1.0" encoding="ISO-8859-1" standalone="yes" ?>
<report>
    <rowsbypage>5</rowsbypage><!--added to complete the solution-->
    <contenido>
        <page>
            <pagenumber>1</pagenumber>
            <row regnum="0">
                <column name="group_id"  type="number">
                    <value><![CDATA[8]]></value>
                    <group_id type="number"><![CDATA[8]]></group_id>
                </column>
                <column name="group_name"  type="text">
                    <value><![CDATA[CTA:Financials]]></value>
                    <group_name type="text"><![CDATA[CTA:Financials]]></group_name>
                </column>
            </row>
            <row regnum="1">
                <column name="group_id"  type="number">
                    <value><![CDATA[9]]></value>
                    <group_id type="number"><![CDATA[9]]></group_id>
                </column>
                <column name="group_name"  type="text">
                    <value><![CDATA[CTA:HR]]></value>
                    <group_name type="text"><![CDATA[CTA: RH]]></group_name>
                </column>
            </row>
            <row regnum="2">
                <column name="group_id"  type="number">
                    <value><![CDATA[7]]></value>
                    <group_id type="number"><![CDATA[7]]></group_id>
                </column>
                <column name="group_name"  type="text">
                    <value><![CDATA[CTA:Accounting]]></value>
                    <group_name type="text"><![CDATA[CTA:Accounting]]></group_name>
                </column>
            </row>
            <row regnum="3">
                <column name="group_id"  type="number">
                    <value><![CDATA[2]]></value>
                    <group_id type="number"><![CDATA[2]]></group_id>
                </column>
                <column name="group_name"  type="text">
                    <value><![CDATA[CTA:Shop]]></value>
                    <group_name type="text"><![CDATA[CTA:Shop]]></group_name>
                </column>
            </row>
            <row regnum="38">
                <column name="group_id"  type="number">
                    <value><![CDATA[3]]></value>
                    <group_id type="number"><![CDATA[3]]></group_id>
                </column>
                <column name="group_name"  type="text">
                    <value><![CDATA[P:Admins]]></value>
                    <group_name type="text"><![CDATA[P:Admins]]></group_name>
                </column>
            </row>
        </page>
        <page>
            <pagenumber>2</pagenumber>
            <row regnum="39">
                <column name="group_id"  type="number">
                    <value><![CDATA[1]]></value>
                    <group_id type="number"><![CDATA[1]]></group_id>
                </column>
                <column name="group_name"  type="text">
                    <value><![CDATA[P:Sys]]></value>
                    <group_name type="text"><![CDATA[P:Sys]]></group_name>
                </column>
            </row>
            <row regnum="40">
                <column name="group_id"  type="number">
                    <value><![CDATA[5]]></value>
                    <group_id type="number"><![CDATA[5]]></group_id>
                </column>
                <column name="group_name"  type="text">
                    <value><![CDATA[P:Dir]]></value>
                    <group_name type="text"><![CDATA[P:Dir]]></group_name>
                </column>
            </row>
            <row regnum="41">
                <column name="group_id"  type="number">
                    <value><![CDATA[6]]></value>
                    <group_id type="number"><![CDATA[6]]></group_id>
                </column>
                <column name="group_name"  type="text">
                    <value><![CDATA[P:Ge]]></value>
                    <group_name type="text"><![CDATA[P:Ge]]></group_name>
                </column>
            </row>
            <row regnum="42">
                <column name="group_id"  type="number">
                    <value><![CDATA[4]]></value>
                    <group_id type="number"><![CDATA[4]]></group_id>
                </column>
                <column name="group_name"  type="text">
                    <value><![CDATA[P:Req]]></value>
                    <group_name type="text"><![CDATA[P:Req]]></group_name>
                </column>
            </row>
        </page>
    </contenido>
</report>
我会感谢你在这方面的帮助,亲切的问候

感谢您在Joel和Michael的帮助下做出的回答。我最后添加了一个标签,标签上按页面显示行数,xsl:

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

<xsl:template match="/" >
        <xsl:apply-templates select="report/contenido/page" />
</xsl:template>

<xsl:template name="dopaging">
        <xsl:param name="pageNumber" />
        <xsl:for-each select="../page/row" >
                <xsl:sort select="column/value" data-type="number"/>
                <xsl:choose>
                        <xsl:when test="position() &gt; ($pageNumber - 1) * /report/rowsbypage and position()  &lt;= $pageNumber * /report/rowsbypage">
                                <xsl:for-each select="column" >
                                <div><xsl:value-of select="value" /></div>
                                </xsl:for-each>
                        </xsl:when>
                </xsl:choose>
        </xsl:for-each>
</xsl:template>

<xsl:template match="pagenumber" >
        <xsl:value-of select="." />
</xsl:template>

<xsl:template match="page">
        <xsl:variable name="pageNumber" select="pagenumber" />
        <div>
                <div>Page <xsl:value-of select="$pageNumber" /></div>
                <xsl:call-template name="dopaging">
                        <xsl:with-param name="pageNumber" select="$pageNumber" />
                </xsl:call-template>
        </div>
</xsl:template>

</xsl:stylesheet>
在Michael的最后一条评论之后,对于每个页面,对所有记录进行一次又一次的排序,最后的XSL如下所示:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" 
xmlns:exsl="http://exslt.org/common" extension-element-prefixes="exsl">
    <xsl:output method="html" encoding="iso-8859-1" version="5.0" />
    <xsl:template match="/" >
            <xsl:variable name="rowsByPage" select="report/rowsbypage" />
            <xsl:variable name="rows">
                    <xsl:for-each select="report/contenido/page/row" >
                            <xsl:sort select="column/group_id" data-type="number" />
                            <div>
                            <xsl:for-each select="column" >
                                    <div><xsl:value-of select="value"/></div>
                            </xsl:for-each>
                            </div>
                    </xsl:for-each>
            </xsl:variable>
            <xsl:for-each select="exsl:node-set($rows)/div[position() mod $rowsByPage = 1]" >
                    <div>
                    <div>Page <xsl:value-of select="position()" /></div>
                    <xsl:for-each select=". | following-sibling::div[position() &lt; $rowsByPage]">
                            <xsl:copy-of select="." />
                    </xsl:for-each>
                    </div>
            </xsl:for-each>
    </xsl:template>

    </xsl:stylesheet>

请尝试下面的代码。我不确定你需要的输出

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

    <xsl:output cdata-section-elements="value group_id group_name"/>

    <xsl:template name="first_group">
        <xsl:variable name="page1_count" select="count(../page[1]/row)"/>
            <xsl:for-each select="../page/row">
                <xsl:sort select="column/value"/>
                <xsl:choose>
                    <xsl:when test="position() &lt;= $page1_count">
                        <xsl:copy-of select="."/>
                    </xsl:when>
                    <xsl:otherwise/>
                </xsl:choose>
            </xsl:for-each>
    </xsl:template>

    <xsl:template name="second_group">
        <xsl:variable name="page1_count" select="count(../page[1]/row)"/>
        <xsl:for-each select="../page/row">
            <xsl:sort select="column/value"/>
            <xsl:choose>
                <xsl:when test="position() &gt; $page1_count">
                    <xsl:copy-of select="."/>
                </xsl:when>
                <xsl:otherwise/>
            </xsl:choose>
        </xsl:for-each>
    </xsl:template>


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


    <xsl:template match="page">
        <xsl:copy>
            <xsl:apply-templates select="pagenumber"/>
            <row>
                <column>Group ID</column>
                <column>Group Name</column>
            </row>
            <xsl:choose>
                <xsl:when test="count(preceding-sibling::page) + 1 = 1">
                    <xsl:call-template name="first_group"/>
                </xsl:when>
                <xsl:otherwise>
                    <xsl:call-template name="second_group"/>
                </xsl:otherwise>
            </xsl:choose>
        </xsl:copy>
    </xsl:template>

</xsl:stylesheet>

组ID
组名
输出在html网格上

我不太清楚“html网格”是什么。下面的样式表提供了XML输出-将其转换为html(表或任何其他结构)应该是相当简单的

这里的主要问题是,您需要分两步进行:首先,对行进行排序;然后将排序的行聚合到页面中,每次5页

XSLT 1.0(需要支持EXSLT node-set()函数:

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:exsl="http://exslt.org/common"
extension-element-prefixes="exsl">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>

<xsl:template match="/">

<xsl:variable name="rows">
    <xsl:for-each select="report/contenido/page/row">
    <xsl:sort select="column/group_id"/>
        <group>
            <id><xsl:value-of select="column/group_id"/></id>
            <name><xsl:value-of select="column/group_name"/></name>
        </group>
    </xsl:for-each>
</xsl:variable>

<output>
    <xsl:for-each select="exsl:node-set($rows)/group[position() mod 5 = 1]">
        <page pagenum="{position()}">
            <xsl:copy-of select=". | following-sibling::group[position() &lt; 5]"/>
        </page>
    </xsl:for-each>
</output>
</xsl:template>

</xsl:stylesheet>

应用于您的输入,结果是:

<?xml version="1.0" encoding="UTF-8"?>
<output>
   <page pagenum="1">
      <group>
         <id>1</id>
         <name>P:Sys</name>
      </group>
      <group>
         <id>2</id>
         <name>CTA:Shop</name>
      </group>
      <group>
         <id>3</id>
         <name>P:Admins</name>
      </group>
      <group>
         <id>4</id>
         <name>P:Req</name>
      </group>
      <group>
         <id>5</id>
         <name>P:Dir</name>
      </group>
   </page>
   <page pagenum="2">
      <group>
         <id>6</id>
         <name>P:Ge</name>
      </group>
      <group>
         <id>7</id>
         <name>CTA:Accounting</name>
      </group>
      <group>
         <id>8</id>
         <name>CTA:Financials</name>
      </group>
      <group>
         <id>9</id>
         <name>CTA: RH</name>
      </group>
   </page>
</output>

1.
P:Sys
2.
CTA:商店
3.
P:管理员
4.
P:Req
5.
P:Dir
6.
P:Ge
7.
CTA:会计
8.
CTA:金融类
9
CTA:RH

1.什么决定了排序后分配给每个组的页码?2.您真的想要如图所示的文本输出吗?1.页码是由一个页面可以包含的行数分配的,2.不,这里的示例很简单,输出在html网格上。在这个示例中,一个页面只能包含5行。IIUC,您的解决方案假设数字为of输入中的页数等于输出所需的页数。这一点从未说明过。请注意,您对每一页的所有行重新排序。我没有对输入数据中的页数发表评论,因为在我最初的帖子中,我没有这些数据,但在您提出问题并看到Joel解决方案后,我注意到需要这样做参数,很容易添加页数,因此我想添加此数据是有效的。关于通过每个页面我对所有行再次排序这一点,是否正确,是否有更好的方法可以做到这一点?我将感谢对此的任何评论,并再次感谢我所遗漏的一切,当我使用您的xsl和我的xml运行xsltproc时,我只是get tag.@user3344938“我是否遗漏了什么”可能,但我不知道是什么。上面的测试是用Xalan、Saxon和libxslt进行的-我相信这就是xsltproc使用的。这工作完美无瑕,我将分析您的XSL,它工作得很好,非常感谢。
<?xml version="1.0" encoding="UTF-8"?>
<output>
   <page pagenum="1">
      <group>
         <id>1</id>
         <name>P:Sys</name>
      </group>
      <group>
         <id>2</id>
         <name>CTA:Shop</name>
      </group>
      <group>
         <id>3</id>
         <name>P:Admins</name>
      </group>
      <group>
         <id>4</id>
         <name>P:Req</name>
      </group>
      <group>
         <id>5</id>
         <name>P:Dir</name>
      </group>
   </page>
   <page pagenum="2">
      <group>
         <id>6</id>
         <name>P:Ge</name>
      </group>
      <group>
         <id>7</id>
         <name>CTA:Accounting</name>
      </group>
      <group>
         <id>8</id>
         <name>CTA:Financials</name>
      </group>
      <group>
         <id>9</id>
         <name>CTA: RH</name>
      </group>
   </page>
</output>