Warning: file_get_contents(/data/phpspider/zhask/data//catemap/0/xml/12.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
使用XSLT1.0从xml中选择不同的节点和总量_Xml_Xslt_Xslt 1.0_Xalan - Fatal编程技术网

使用XSLT1.0从xml中选择不同的节点和总量

使用XSLT1.0从xml中选择不同的节点和总量,xml,xslt,xslt-1.0,xalan,Xml,Xslt,Xslt 1.0,Xalan,我有XML,它应该通过XSLT1.0进行转换。XML文件夹“字段”定义每个“行”元素的名称顺序。因此,每个“行”文件夹中的MaterialCode都位于第一位,StorageMatCode位于第二位,“Amount”位于第三位。我需要删除所有“MaterialCode”副本,但将所有“金额”放在一个中。 输入xml: <Response xmlns="http://www.sample.ru/sample/" xmlns:xsi="http://www.w3.org/2001/XMLSch

我有XML,它应该通过XSLT1.0进行转换。XML文件夹“字段”定义每个“行”元素的名称顺序。因此,每个“行”文件夹中的MaterialCode都位于第一位,StorageMatCode位于第二位,“Amount”位于第三位。我需要删除所有“MaterialCode”副本,但将所有“金额”放在一个中。 输入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 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>
<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>

储物垫
1.0.0
材料代码
StorageMatCode
数量
475625947
456789
1000
804685387
273456
3047
973681347
578357
2037
804685387
273456
5000
我想获取此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 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>
<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>

475625947
1000
804685387
8047
973681347
2037
我创建了这个XSLT:

<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 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>
<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 Processor for 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中的节点

如果您使用的是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中的节点