2017-02-03 13 views
0

XMLがあります。XSLT 1.0で変換する必要があります。 XMLフォルダ「フィールド」は、すべての「行」要素の名前順を定義します。したがって、すべての "行"フォルダ内のMaterialCodeは最初の位置にあり、StorageMatCodeは2番目に、 "金額"は3番目の位置にあります。すべての "MaterialCode"重複を削除する必要がありますが、すべての "Amounts"を1つにまとめてください。 入力のxml:xslt 1.0を使用して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、私のシステムはXSLTのためにApache Xalan Processorを使用します。 XSLTからこの行を拒否する:

私は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が正しく

関連する問題