Как группировать и суммировать значения в XSLT
Для каждого "агентства" node мне нужно найти элементы "stmt" , которые имеют одинаковые значения ключа1, key2, key3 и выводят только один "stmt" node с значениями "comm" и "prem" суммируются вместе. Для любых элементов "stmt" внутри этого "агентства", которые не соответствуют каким-либо другим элементам "stmt" на основе key1, key2 и key3, мне нужно вывести их как есть. Поэтому после трансформации первое "агентство" node будет иметь только два узла "stmt" (один из них суммирован), а второе "агентство" node будет передано так же, как и потому, что ключи не совпадают. Решения 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>
Ответы
Ответ 1
И решение 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>
Ответ 2
В 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>
, когда применяется к предоставленному 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>
Ответ 3
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:msxsl="urn:schemas-microsoft-com:xslt" exclude-result-prefixes="msxsl">
<xsl:output method="xml" indent="yes"/>
<xsl:template match="/|*">
<xsl:copy>
<xsl:apply-templates select="*" />
</xsl:copy>
</xsl:template>
<xsl:template match="stmt">
<xsl:variable name="stmtGroup" select="../stmt[(key1=current()/key1) and (key2=current()/key2) and (key3=current()/key3)]" />
<xsl:if test="generate-id()=generate-id($stmtGroup[1])">
<xsl:copy>
<key1>
<xsl:value-of select="key1"/>
</key1>
<key2>
<xsl:value-of select="key2"/>
</key2>
<key3>
<xsl:value-of select="key3"/>
</key3>
<comm>
<xsl:value-of select="format-number(sum($stmtGroup/comm), '#.00')"/>
</comm>
<prem>
<xsl:value-of select="format-number(sum($stmtGroup/prem), '#.00')"/>
</prem>
</xsl:copy>
</xsl:if>
</xsl:template>
</xsl:stylesheet>