对于每个"代理"节点,我需要找到具有相同key1,key2,key3值的"stmt"元素,并输出一个"stmt"节点,其中"comm"和"prem"值相加在一起.对于该"代理"中的任何"stmt"元素,它们与基于key1,key2和key3的任何其他"stmt"元素都不匹配,我需要按原样输出它们.因此,在转换之后,第一个"代理"节点将只有两个"stmt"节点(一个求和),而第二个"代理"节点将按原样传递,因为密钥不匹配.XSLT 1.0或2.0解决方案是可以的......虽然我的样式表目前是1.0.请注意,代理节点可以具有任意数量的"stmt"元素,这些元素具有需要分组和求和的匹配键以及不需要的任何数字.
<statement>
<agency>
<stmt>
<key1>1234</key1>
<key2>ABC</key2>
<key3>15.000</key3>
<comm>75.00</comm>
<prem>100.00</prem>
</stmt>
<stmt>
<key1>1234</key1>
<key2>ABC</key2>
<key3>15.000</key3>
<comm>25.00</comm>
<prem>200.00</prem>
</stmt>
<stmt>
<key1>1234</key1>
<key2>ABC</key2>
<key3>17.50</key3>
<comm>25.00</comm>
<prem>100.00</prem>
</stmt>
</agency>
<agency>
<stmt>
<key1>5678</key1>
<key2>DEF</key2>
<key3>15.000</key3>
<comm>10.00</comm>
<prem>20.00</prem>
</stmt>
<stmt>
<key1>5678</key1>
<key2>DEF</key2>
<key3>17.000</key3>
<comm>15.00</comm>
<prem>12.00</prem>
</stmt>
</agency>
Run Code Online (Sandbox Code Playgroud)
Dim*_*hev 13
和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"
exclude-result-prefixes="xs"
>
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:template match="node()|@*">
<xsl:copy>
<xsl:apply-templates select="node()|@*"/>
</xsl:copy>
</xsl:template>
<xsl:template match="agency">
<agency>
<xsl:for-each-group select="stmt" group-by=
"concat(key1, '+', key2, '+', key3)">
<stmt>
<xsl:copy-of select=
"current-group()[1]/*[starts-with(name(),'key')]"/>
<comm>
<xsl:value-of select="sum(current-group()/comm)"/>
</comm>
<prem>
<xsl:value-of select="sum(current-group()/prem)"/>
</prem>
</stmt>
</xsl:for-each-group>
</agency>
</xsl:template>
</xsl:stylesheet>
Run Code Online (Sandbox Code Playgroud)
在XSLT 1.0中,使用Muenchian方法进行分组(使用复合键).
这种转变:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:key name="kStmtByKeys" match="stmt"
use="concat(generate-id(..), key1, '+', key2, '+', key3)"/>
<xsl:template match="node()|@*">
<xsl:copy>
<xsl:apply-templates select="node()|@*"/>
</xsl:copy>
</xsl:template>
<xsl:template match="agency">
<agency>
<xsl:for-each select=
"stmt[generate-id()
=
generate-id(key('kStmtByKeys',
concat(generate-id(..), key1, '+', key2, '+', key3)
)[1]
)
]
">
<xsl:variable name="vkeyGroup" select=
"key('kStmtByKeys', concat(generate-id(..), key1, '+', key2, '+', key3))"/>
<stmt>
<xsl:copy-of select="*[starts-with(name(), 'key')]"/>
<comm>
<xsl:value-of select="sum($vkeyGroup/comm)"/>
</comm>
<prem>
<xsl:value-of select="sum($vkeyGroup/prem)"/>
</prem>
</stmt>
</xsl:for-each>
</agency>
</xsl:template>
</xsl:stylesheet>
Run Code Online (Sandbox Code Playgroud)
当应用于提供的XML文档时,产生想要的结果:
<statement>
<agency>
<stmt>
<key1>1234</key1>
<key2>ABC</key2>
<key3>15.000</key3>
<comm>100</comm>
<prem>300</prem>
</stmt>
<stmt>
<key1>1234</key1>
<key2>ABC</key2>
<key3>17.50</key3>
<comm>25</comm>
<prem>100</prem>
</stmt>
</agency>
<agency>
<stmt>
<key1>5678</key1>
<key2>DEF</key2>
<key3>15.000</key3>
<comm>10</comm>
<prem>20</prem>
</stmt>
<stmt>
<key1>5678</key1>
<key2>DEF</key2>
<key3>17.000</key3>
<comm>15</comm>
<prem>12</prem>
</stmt>
</agency>
</statement>
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
33392 次 |
| 最近记录: |