我有XML,应该由XSLT 1.0转换。XML 文件夹"字段"定义每个"行"元素的名称顺序。因此,每个"行"文件夹中的MaterialCode具有第一个位置,StorageMatCode位于第二位,"Amount"位于第三位。我需要删除所有"材料代码"重复项,但将所有"金额"放在一个中。输入 xml:
<Response xmlns="http://www.sample.ru/sample/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<Header>
<ObjectType>StorageMats</ObjectType>
<Version>1.0.0</Version>
<Fields>
<Field type="decimal">MaterialCode</Field>
<Field type="decimal">StorageMatCode</Field>
<Field type="decimal">Amount</Field>
</Fields>
</Header>
<Body>
<Row>
<FieldValue>475625947</FieldValue>
<FieldValue>456789</FieldValue>
<FieldValue>1000</FieldValue>
</Row>
<Row>
<FieldValue>804685387</FieldValue>
<FieldValue>273456</FieldValue>
<FieldValue>3047</FieldValue>
</Row>
<Row>
<FieldValue>973681347</FieldValue>
<FieldValue>578357</FieldValue>
<FieldValue>2037</FieldValue>
</Row>
<Row>
<FieldValue>804685387</FieldValue>
<FieldValue>273456</FieldValue>
<FieldValue>5000</FieldValue>
</Row>
</Body>
</Response>
我想获取此 XML:
<?xml version="1.0" encoding="UTF-8"?>
<BDStorageMats xmlns="http://www.sample.ru/sample/BDStorageMats/1.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<BDStorageMat>
<MaterialCode>475625947</MaterialCode>
<Amount>1000</Amount>
</BDStorageMat>
<BDStorageMat>
<MaterialCode>804685387</MaterialCode>
<Amount>8047</Amount>
</BDStorageMat>
<BDStorageMat>
<MaterialCode>973681347</MaterialCode>
<Amount>2037</Amount>
</BDStorageMat>
</BDStorageMats>
我创建了这个 XSLT:
<xsl:stylesheet version="1.0" xmlns="http://www.sample.ru/sample/BDStorageMats/1.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output encoding="UTF-8" indent="yes" method="xml" version="1.0"/>
<xsl:key match="*[local-name()= 'Response']/*[local-name()= 'Body']/*[local-name()= 'Row']" name="codeDistinct" use="*[local-name()= 'FieldValue'][count(*[local-name()= 'Response']/*[local-name()= 'Header']/*[local-name()= 'Fields']/*[local-name()= 'Field'][.='MaterialCode']/preceding-sibling::*)+1]"/>
<xsl:template match="/">
<BDStorageMats>
<xsl:variable name="amountPosition" select="count(*[local-name()= 'Response']/*[local-name()= 'Header']/*[local-name()= 'Fields']/*[local-name()= 'Field'][.='Amount']/preceding-sibling::*)+1"/>
<xsl:variable name="materialCodePosition" select="count(*[local-name()= 'Response']/*[local-name()= 'Header']/*[local-name()= 'Fields']/*[local-name()= 'Field'][.='MaterialCode']/preceding-sibling::*)+1"/>
<xsl:for-each select="*[local-name()= 'Response']/*[local-name()= 'Body']/*[local-name()= 'Row'][generate-id() = generate-id(key('codeDistinct', *[local-name()= 'FieldValue'][count(*[local-name()= 'Response']/*[local-name()= 'Header']/*[local-name()= 'Fields']/*[local-name()= 'Field'][.='MaterialCode']/preceding-sibling::*)+1]))[1]]">
<xsl:variable name="keyGroup" select="key('codeDistinct', *[local-name()= 'FieldValue'][count(*[local-name()= 'Response']/*[local-name()= 'Header']/*[local-name()= 'Fields']/*[local-name()= 'Field'][.='MaterialCode']/preceding-sibling::*)+1])"/>
<BDStorageMat>
<MaterialCode>
<xsl:value-of select="(*[local-name()= 'FieldValue'])[$materialCodePosition]"/>
</MaterialCode>
<Amount>
<xsl:value-of select="sum($keyGroup/(*[local-name()= 'FieldValue'])[$amountPosition])"/>
</Amount>
</BDStorageMat>
</xsl:for-each>
</BDStorageMats>
</xsl:template>
</xsl:stylesheet>
它在 Altova 中工作正常,但我的系统使用 Apache Xalan 处理器进行 XSLT,它拒绝了 XSLT 中的这一行:
<Amount>
<xsl:value-of select="sum($keyGroup/(*[local-name()= 'FieldValue']) [$amountPosition])"/>
</Amount>
还有其他方法可以通过 XSLT 1.0 执行我想要的操作吗?
如果您使用的是 Xalan(或其他支持 EXSLT set:distinct()
扩展功能的处理器),您可以执行以下操作:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:ns1="http://www.sample.ru/sample/"
xmlns:set="http://exslt.org/sets"
exclude-result-prefixes="ns1 set">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<!-- identity transform -->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:key name="row" match="ns1:Row" use="ns1:FieldValue[1]" />
<xsl:template match="/ns1:Response">
<BDStorageMats xmlns="http://www.sample.ru/sample/BDStorageMats/1.0">
<xsl:for-each select="set:distinct(ns1:Body/ns1:Row/ns1:FieldValue[1])">
<BDStorageMat>
<MaterialCode>
<xsl:value-of select="." />
</MaterialCode>
<Amount>
<xsl:value-of select="sum(key('row', .)/ns1:FieldValue[3])" />
</Amount>
</BDStorageMat>
</xsl:for-each>
</BDStorageMats>
</xsl:template>
</xsl:stylesheet>
请注意使用前缀来正确选择输入 XML 中的节点。