我有一个xml文档,其中包含几个测试定义块。使用xsl转换,我想将这些测试定义块中的一些合并在一起,当它们具有相同的名称时。 合并后,我想为每个合并的测试定义分配一个数字(Order)。
示例输入: (订单在任何地方都是0是故意的,因为我现在还不知道,在合并之后还会有多少次测试)
<TestDef>
<Name>Test1</Name>
<Order>0</Order>
<Results>
<Result Name="Result1"/>
<Result Name="Result2"/>
</Results>
</TestDef>
<TestDef>
<Name>Test2</Name>
<Order>0</Order>
<Results>
<Result Name="Result5"/>
<Result Name="Result6"/>
</Results>
</TestDef>
<TestDef>
<Name>Test1</Name>
<Order>0</Order>
<Results>
<Result Name="Result3"/>
<Result Name="Result4"/>
</Results>
</TestDef>
通缉输出: (现在每个合并测试都有一个分配的订单号,从1开始,没有间隙计数)
<TestDef>
<Name>Test1</Name>
<Order>1</Order>
<Results>
<Result Name="Result1"/>
<Result Name="Result2"/>
<Result Name="Result3"/>
<Result Name="Result4"/>
</Results>
</TestDef>
<TestDef>
<Name>Test2</Name>
<Order>2</Order>
<Results>
<Result Name="Result5"/>
<Result Name="Result6"/>
</Results>
</TestDef>
当前输出: (Test2的Order标签是重点)
<TestDef>
<Name>Test1</Name>
<Order>1</Order>
<Results>
<Result Name="Result1"/>
<Result Name="Result2"/>
<Result Name="Result3"/>
<Result Name="Result4"/>
</Results>
</TestDef>
<TestDef>
<Name>Test2</Name>
<Order>3</Order>
<Results>
<Result Name="Result5"/>
<Result Name="Result6"/>
</Results>
</TestDef>
我当前的xslt:
<xsl:key name="unique-tests" match="TestDef" use="Name" />
<xsl:strip-space elements="*"/>
<xsl:template match="@* | node()">
<xsl:copy>
<xsl:apply-templates select="@* | node()"/>
</xsl:copy>
</xsl:template>
<!-- Merge all tests with the same name together and add all results to
the one new test
This grouping was done, using the Muenchian Method:
http://www.jenitennison.com/xslt/grouping/muenchian.html -->
<xsl:template match="TestDef">
<xsl:if test="generate-id()= generate-id(key('unique-tests', Name))">
<Name>
<xsl:value-of select="Name" />
</Name>
<Order>
<xsl:number />
</Order>
<Results>
<xsl:for-each select="key('unique-tests', Name)">
<xsl:copy-of select="Results/node()"/>
</xsl:for-each>
</Results>
</xsl:if>
</xsl:template>
问题:
分组工作得很好,但订单编号只是以我想要的方式工作。当我在显示的xslt中使用<xsl:number/>
时,他将从输入中计算每个测试,但只输出一些数字。我的示例的输出可能类似于Test1(order = 1)和Test2(order = 3)。
当我在if条件之前添加<xsl:number/>
的附加输出时,他将按顺序输出每个数字,即使测试定义本身不会在输出中出现。
现在我看到了其他类似的问题:xsl counter with and condition
在那里他们计算条件,但由于我自己的条件不能表示为XPath,我不能将它用于count
的{{1}}属性
所以,我希望订单标签的数量从1到输出中的任何数量的TestDef,没有任何间隙。我该如何做到这一点?
答案 0 :(得分:1)
恕我直言,最好是有选择地应用模板 - 即仅应用于组中第一个TestDef
元素 - 然后使用position()
函数对它们进行编号。
为了演示,我们必须首先拥有一个带有单个根元素的有效XML输入,例如:
<强> XML 强>
<root>
<TestDef>
<Name>Test1</Name>
<Order>0</Order>
<Results>
<Result Name="Result1"/>
<Result Name="Result2"/>
</Results>
</TestDef>
<TestDef>
<Name>Test2</Name>
<Order>0</Order>
<Results>
<Result Name="Result5"/>
<Result Name="Result6"/>
</Results>
</TestDef>
<TestDef>
<Name>Test1</Name>
<Order>0</Order>
<Results>
<Result Name="Result3"/>
<Result Name="Result4"/>
</Results>
</TestDef>
</root>
然后你可以这样做:
XSLT 1.0
<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:strip-space elements="*"/>
<xsl:key name="unique-tests" match="TestDef" use="Name" />
<!-- identity transform -->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="/root">
<xsl:copy>
<xsl:apply-templates select="TestDef[generate-id() = generate-id(key('unique-tests', Name)[1])]"/>
</xsl:copy>
</xsl:template>
<xsl:template match="TestDef">
<xsl:copy>
<Name><xsl:value-of select="Name" /></Name>
<Order><xsl:value-of select="position()" /></Order>
<Results>
<xsl:for-each select="key('unique-tests', Name)">
<xsl:copy-of select="Results/node()"/>
</xsl:for-each>
</Results>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
或者,您也可以使用以下内容将条件复制到xsl:number
<xsl:number count="TestDef[generate-id() = generate-id(key('unique-tests', Name)[1])]"/>