使用XSLT在XML中对表元素进行分组

时间:2016-05-05 15:57:36

标签: xml xslt xml-parsing xslt-2.0 xslt-grouping

我已经在下面的转换代码中工作了很长时间。我不确定XSLT& S中的要求是否可行。需要你的帮助来确定它是否可能使用XSLT。

我的输入XML如下,

<S:Envelope xmlns:S="http://schemas.xmlsoap.org/soap/envelope/">
   <S:Body>
      <base:callFlowResponse xmlns:base="http://schema/webservices.elink.solcorp.com">
         <Messages>
            <More/>
         </Messages>
         <OUTPUT>
               <MIR-DV-INT>100</MIR-DV-INT>
               <MIR-DEP-AMT>0000000000042832.50</MIR-DEP-AMT>
               <MIR-DV-TOTAL-AMT-DUE>1000</MIR-DV-TOTAL-AMT-DUE>
               <MIR-MAX-REM-TOP-PREM>1000</MIR-MAX-REM-TOP-PREM>
               <MIR-DV-DUE-DT-G>
                  <TableMIR-DV-DUE-DT-T>
                     <MIR-DV-DUE-DT-T>2011-01-01</MIR-DV-DUE-DT-T>
                     <MIR-DV-DUE-DT-T/>
                     <MIR-DV-DUE-DT-T/>
                  </TableMIR-DV-DUE-DT-T>
               </MIR-DV-DUE-DT-G>
               <MIR-DV-POL-MPREM-AMT-G>
                  <TableMIR-DV-POL-MPREM-AMT-T>
                     <MIR-DV-POL-MPREM-AMT-T>1000</MIR-DV-POL-MPREM-AMT-T>
                     <MIR-DV-POL-MPREM-AMT-T/>
                     <MIR-DV-POL-MPREM-AMT-T/>
                  </TableMIR-DV-POL-MPREM-AMT-T>
               </MIR-DV-POL-MPREM-AMT-G>
               <MIR-DV-BASIC-SERV-TAX-AMT-G>
                  <TableMIR-DV-BASIC-SERV-TAX-AMT-T>
                     <MIR-DV-BASIC-SERV-TAX-AMT-T>10000</MIR-DV-BASIC-SERV-TAX-AMT-T>
                     <MIR-DV-BASIC-SERV-TAX-AMT-T/>
                     <MIR-DV-BASIC-SERV-TAX-AMT-T/>
                  </TableMIR-DV-BASIC-SERV-TAX-AMT-T>
               </MIR-DV-BASIC-SERV-TAX-AMT-G>
               <MIR-DV-TOTL-SW-CESS-AMT-G>
                  <TableMIR-DV-TOTL-SW-CESS-AMT-T>
                     <MIR-DV-TOTL-SW-CESS-AMT-T>1000</MIR-DV-TOTL-SW-CESS-AMT-T>
                     <MIR-DV-TOTL-SW-CESS-AMT-T/>
                     <MIR-DV-TOTL-SW-CESS-AMT-T/>
                  </TableMIR-DV-TOTL-SW-CESS-AMT-T>
               </MIR-DV-TOTL-SW-CESS-AMT-G>
               <MIR-DV-KR-KL-CESS-AMT-G>
                  <TableMIR-DV-KR-KL-CESS-AMT-T>
                     <MIR-DV-KR-KL-CESS-AMT-T>1000</MIR-DV-KR-KL-CESS-AMT-T>
                     <MIR-DV-KR-KL-CESS-AMT-T/>
                     <MIR-DV-KR-KL-CESS-AMT-T/>
                  </TableMIR-DV-KR-KL-CESS-AMT-T>
               </MIR-DV-KR-KL-CESS-AMT-G>
               <MIR-DV-PREM-AMT-G>
                  <TableMIR-DV-PREM-AMT-T>
                     <MIR-DV-PREM-AMT-T>10000</MIR-DV-PREM-AMT-T>
                     <MIR-DV-PREM-AMT-T/>
                     <MIR-DV-PREM-AMT-T/>
                  </TableMIR-DV-PREM-AMT-T>
               </MIR-DV-PREM-AMT-G>
               <MIR-DV-ADV-PREM-AMT>1000</MIR-DV-ADV-PREM-AMT>
               <MIR-TOTL-SW-BH-CESS-TAX-AMT>1000</MIR-TOTL-SW-BH-CESS-TAX-AMT>
               <MIR-KR-KL-CESS-TAX-AMT>1000</MIR-KR-KL-CESS-TAX-AMT>
           </OUTPUT>
      </base:callFlowResponse>
   </S:Body>
</S:Envelope>

输出XML应为as,

<S:Envelope xmlns:S="http://schemas.xmlsoap.org/soap/envelope/">
   <S:Body>
      <base:callFlowResponse xmlns:base="http://schema/webservices.elink.solcorp.com">
         <Messages>
            <More/>
         </Messages>
         <OUTPUT>
               <MIR-DV-INT>100</MIR-DV-INT>
               <MIR-DEP-AMT>0000000000042832.50</MIR-DEP-AMT>
               <MIR-DV-TOTAL-AMT-DUE>1000</MIR-DV-TOTAL-AMT-DUE>
               <MIR-MAX-REM-TOP-PREM>1000</MIR-MAX-REM-TOP-PREM>
               <GROUP-1>
                     <MIR-DV-DUE-DT-T>2011-01-01</MIR-DV-DUE-DT-T>
                     <MIR-DV-POL-MPREM-AMT-T>1000</MIR-DV-POL-MPREM-AMT-T>
                     <MIR-DV-BASIC-SERV-TAX-AMT-T>10000</MIR-DV-BASIC-SERV-TAX-AMT-T>
                     <MIR-DV-TOTL-SW-CESS-AMT-T>1000</MIR-DV-TOTL-SW-CESS-AMT-T>
                     <MIR-DV-KR-KL-CESS-AMT-T>1000</MIR-DV-KR-KL-CESS-AMT-T>
                     <MIR-DV-PREM-AMT-T>10000</MIR-DV-PREM-AMT-T>
               </GROUP-1>
            <GROUP-2>
                     <MIR-DV-DUE-DT-T></MIR-DV-DUE-DT-T>
                     <MIR-DV-POL-MPREM-AMT-T></MIR-DV-POL-MPREM-AMT-T>
                     <MIR-DV-BASIC-SERV-TAX-AMT-T></MIR-DV-BASIC-SERV-TAX-AMT-T>
                     <MIR-DV-TOTL-SW-CESS-AMT-T></MIR-DV-TOTL-SW-CESS-AMT-T>
                     <MIR-DV-KR-KL-CESS-AMT-T></MIR-DV-KR-KL-CESS-AMT-T>
                     <MIR-DV-PREM-AMT-T></MIR-DV-PREM-AMT-T>
               </GROUP-2>               
            <GROUP-3>
                     <MIR-DV-DUE-DT-T></MIR-DV-DUE-DT-T>
                     <MIR-DV-POL-MPREM-AMT-T></MIR-DV-POL-MPREM-AMT-T>
                     <MIR-DV-BASIC-SERV-TAX-AMT-T></MIR-DV-BASIC-SERV-TAX-AMT-T>
                     <MIR-DV-TOTL-SW-CESS-AMT-T></MIR-DV-TOTL-SW-CESS-AMT-T>
                     <MIR-DV-KR-KL-CESS-AMT-T></MIR-DV-KR-KL-CESS-AMT-T>
                     <MIR-DV-PREM-AMT-T></MIR-DV-PREM-AMT-T>
               </GROUP-3>  
               <MIR-DV-ADV-PREM-AMT>1000</MIR-DV-ADV-PREM-AMT>
               <MIR-TOTL-SW-BH-CESS-TAX-AMT>1000</MIR-TOTL-SW-BH-CESS-TAX-AMT>
               <MIR-KR-KL-CESS-TAX-AMT>1000</MIR-KR-KL-CESS-TAX-AMT>
           </OUTPUT>
      </base:callFlowResponse>
   </S:Body>
</S:Envelope>

我已经看到了在分散节点上进行分组的各种示例。但我的要求是不同的。我有六个表,所有相关的值都像数组一样给出。我必须逐个元素地阅读所有六个表格。创建一个组。我应该这样做,直到读取所有节点(在这种情况下,大小为3)

如果你至少可以用一个类似的例子来指导我,那将是一个很大的帮助。我可以参考我的编码。

感谢您的帮助。

1 个答案:

答案 0 :(得分:1)

这是一个处理第一个“table”子节点并在每个group wrapper元素中收集兄弟表子节点的示例:

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    exclude-result-prefixes="xs"
    version="2.0">

    <xsl:output indent="yes"/>

    <xsl:template match="@* | node()">
        <xsl:copy>
            <xsl:apply-templates select="@* , node()"/>
        </xsl:copy>
    </xsl:template>

    <xsl:template match="OUTPUT/*[*[starts-with(local-name(), 'Table')]][1]">
        <xsl:apply-templates select="*/*" mode="group"/>
    </xsl:template>

    <xsl:template match="OUTPUT/*[*[starts-with(local-name(), 'Table')]][position() gt 1]"/>

    <xsl:template match="*" mode="group">
        <xsl:variable name="pos" select="position()"/>
        <xsl:element name="group-{position()}">
            <xsl:copy-of select="., ../../following-sibling::*[*[starts-with(local-name(), 'Table')]]/*/*[$pos]"/>
        </xsl:element>
    </xsl:template>
</xsl:stylesheet>

作为收集兄弟元素的替代方法,您可以使用位置分组:

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    exclude-result-prefixes="xs"
    version="2.0">

    <xsl:output indent="yes"/>

    <xsl:template match="@* | node()">
        <xsl:copy>
            <xsl:apply-templates select="@* , node()"/>
        </xsl:copy>
    </xsl:template>

    <xsl:template match="OUTPUT">
        <xsl:copy>
            <xsl:for-each-group select="*" group-adjacent="boolean(self::*[*[starts-with(local-name(), 'Table')]])">
                <xsl:choose>
                    <xsl:when test="current-grouping-key()">
                        <xsl:variable name="row-count" select="count(*[1]/*)"/>
                        <xsl:for-each-group select="current-group()/*/*" group-by="position() mod $row-count">
                            <xsl:element name="GROUP-{position()}">
                                <xsl:copy-of select="current-group()"/>
                            </xsl:element>
                        </xsl:for-each-group>
                    </xsl:when>
                    <xsl:otherwise>
                        <xsl:copy-of select="current-group()"/>
                    </xsl:otherwise>
                </xsl:choose>
            </xsl:for-each-group>
        </xsl:copy>
    </xsl:template>

</xsl:stylesheet>