2017-02-03 61 views
0

我有XML,它应该由XSLT 1.0进行转换。 XML文件夹“Fields”为每个“Row”元素定义名称顺序。因此,每个“行”文件夹中的MaterialCode都有第一个位置,StorageMatCode是第二个,“Amount”是第三个。我需要删除所有“MaterialCode”重复项,但将所有“数量”都放在一个中。 输入XML:使用xslt从xml中选择不同的节点和总量1.0

<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 a nd它拒绝从XSLT这条线:

<Amount> 
    <xsl:value-of select="sum($keyGroup/(*[local-name()= 'FieldValue']) [$amountPosition])"/> 
</Amount> 

是否有任何其他方式来做我想通过XSLT 1.0?

回答

2

如果您使用的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正确。