如何使用xslt对xml文件中的某些节点进行分组和删除

时间:2012-10-12 07:52:39

标签: xslt xslt-1.0

我有这个xml文件:

<?xml version="1.0" encoding="UTF-8"?>
<o-com-inter xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="1">
<rules deplVer="125">
<cre-det-r name="Auto_R_1">
    <act>false</act>
    <thres-curr>000</thres-curr>
    <thress>
        <thres name="Auto_R_125_1">
            <sco>11111</sco>
            <serv-res>
                <serv-res>ABC</serv-res>
            </serv-res>
            <cat>
                <cate>CatA</cate>
            </cat>
            <o-ru-con>
                <co-ru-con b-name="InterMge" name="ac_T_EQ_000">
                    <si-ru-con>
                        <ru-con-ite>
                            <ru-con-ite-lit>
                                <nae1>ac</nae1>
                                <ope>T_EQ</ope>
                                <val1>000</val1>
                            </ru-con-ite-lit>
                            <bId>2</bId>
                        </ru-con-ite>
                    </si-ru-con>
                </co-ru-con>
                <co-ru-con b-name="InterMge" name="ac_T_EQ_001">
                    <si-ru-con>
                        <ru-con-ite>
                            <ru-con-ite-lit>
                                <nae1>ac</nae1>
                                <ope>T_EQ</ope>
                                <val1>001</val1>
                            </ru-con-ite-lit>
                            <bId>2</bId>
                        </ru-con-ite>
                    </si-ru-con>
                </co-ru-con>
                <co-ru-con b-name="InterMge" name="tra_T_EQ_014">
                    <si-ru-con>
                        <ru-con-ite>
                            <ru-con-ite-lit>
                                <nae1>tra</nae1>
                                <ope>T_EQ</ope>
                                <val1>014</val1>
                            </ru-con-ite-lit>
                            <bId>3</bId>
                        </ru-con-ite>
                    </si-ru-con>
                </co-ru-con>
                <co-ru-con b-name="InterMge" name="tra_T_EQ_015">
                    <si-ru-con>
                        <ru-con-ite>
                            <ru-con-ite-lit>
                                <nae1>tra</nae1>
                                <ope>T_EQ</ope>
                                <val1>015</val1>
                            </ru-con-ite-lit>
                            <bId>3</bId>
                        </ru-con-ite>
                    </si-ru-con>
                </co-ru-con>
            </o-ru-con>
            <al>true</al>
        </thres>
    </thress>
    <description> rSId=125 Fp=0.1234567 Dr=1.0 
        rSId=48 SFp=1.0 SDr=1.0_2012-10-10T09:55:09+02:00
    </description>
  </cre-det-r>
  <cre-det-r>
    ....If this block exists, it will contain the same elements as above in the same order, otherwise, this block doesn't exists and the code contains only the preceding "cre-det-r" block.
  </cre-det-r>
 </rules>
</o-com-inter>

我想检查每个 bId 元素的值。如果此值与 o-ru-con 元素中的前一个或后一个 bId 元素相同,那么我将对所有不同的 co-ru-进行分组con 阻止(在整个块 o-ru-con 中),在 co-ru-con bId 值>如预期的输出文件所示。 我试图使用这个xsl文件:

<xsl:stylesheet 
 version="1.0"
 xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

<xsl:strip-space elements="*"/>
<xsl:output indent="yes"/>
<!--  -->
<xsl:template match="*">
<xsl:copy>
    <xsl:copy-of select="@*"/>
    <xsl:apply-templates/>
</xsl:copy>
</xsl:template> 

<xsl:key name="k1"
match="ru-con-ite-lit[bId = preceding-sibling::ru-con-ite-lit[1]/bId]"
use="generate-id(preceding-sibling::ru-con-ite-lit[not(bId = preceding-sibling::ru-con-ite-lit[1]/bId)][1])"/>

<xsl:template match="ru-con-ite">
<xsl:copy>
    <xsl:copy-of select="@*"/>
    <xsl:apply-templates select="ru-con-ite-lit[not(preceding-sibling::ru-con-ite-lit[1]) or not(bId = preceding-sibling::ru-con-ite-lit[1]/bId)]"/>
  </xsl:copy>
</xsl:template>

<xsl:template match="ru-con-ite-lit">
<xsl:copy>
    <xsl:copy-of select="@*"/>
 <!--     <xsl:apply-templates/>-->

    <xsl:apply-templates select=". | key('k1', generate-id())" mode="sp"/>

 </xsl:copy>
</xsl:template>

<xsl:template match="ru-con-ite-lit" mode="sp">

    <xsl:copy-of select="node()[not(self::bId)]"/>

 </xsl:template>

</xsl:stylesheet>

我可以使用此xsl文件删除xml文件中的元素 bId ,但我无法将具有相同值的不同 co-ru-con 块分组在一个 co-ru-con 块中的 bId 。预期的输出将如下所示:

<?xml version="1.0" encoding="UTF-8"?>
<o-com-inter xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="1">
<rules deplVer="125">
<cre-det-r name="Auto_R_1">
    <act>false</act>
    <thres-curr>000</thres-curr>
    <thress>
        <thres name="Auto_R_125_1">
            <sco>11111</sco>
            <serv-res>
                <serv-res>ABC</serv-res>
            </serv-res>
            <cat>
                <cate>CatA</cate>
            </cat>
            <o-ru-con>
                <co-ru-con b-name="InterMge" name="bId2"><!-- the value of name here has been changed by the tag-name "bId" and the value 2 of this one -->
                    <si-ru-con>
                        <ru-con-ite>
                            <ru-con-ite-lit>
                                <nae1>ac</nae1>
                                <ope>T_EQ</ope>
                                <val1>000</val1>
                            </ru-con-ite-lit>
                            <!-- Here was the element <bId> with the value 2 and this has been deleted after the transformation -->
                        </ru-con-ite>
                    </si-ru-con>
                    <si-ru-con>
                        <ru-con-ite>
                            <ru-con-ite-lit>
                                <nae1>ac</nae1>
                                <ope>T_EQ</ope>
                                <val1>001</val1>
                            </ru-con-ite-lit>
                            <!-- Here was the element <bId> with the value 2 and this has been deleted after the transformation -->
                        </ru-con-ite>
                    </si-ru-con>
                </co-ru-con>
                <co-ru-con b-name="InterMge" name="bId3"><!-- the value of name here has been changed by the tag-name "bId" and the value 3 of this one -->
                    <si-ru-con>
                        <ru-con-ite>
                            <ru-con-ite-lit>
                                <nae1>tra</nae1>
                                <ope>T_EQ</ope>
                                <val1>014</val1>
                            </ru-con-ite-lit>
                            <!-- Here was the element <bId> with the value 3and this has been deleted after the transformation -->
                        </ru-con-ite>
                    </si-ru-con>
                    <si-ru-con>
                        <ru-con-ite>
                            <ru-con-ite-lit>
                                <nae1>tra</nae1>
                                <ope>T_EQ</ope>
                                <val1>015</val1>
                            </ru-con-ite-lit>
                            <!-- Here was the element <bId> with thw value 3 and this has been deleted after the transformation -->
                        </ru-con-ite>
                    </si-ru-con>
                </co-ru-con>
            </o-ru-con>
            <al>true</al>
        </thres>
    </thress>
    <description> rSId=125 Fp=0.1234567 Dr=1.0 
        rSId=48 SFp=1.0 SDr=1.0_2012-10-10T09:55:09+02:00
    </description>
</cre-det-r>
<cre-det-r>
    ....If this block exists, it will contain the same elements as above in the same order, otherwise, this block doesn't exists and the code contains only the preceding "cre-det-r" block.
</cre-det-r>

   

由于

1 个答案:

答案 0 :(得分:2)

这看起来像是一个分组问题,在XSLT1.0中,这是Muenchian Grouping的工作。在您的情况下,您通过父 o-ru-con 元素及其后代 bId 的组合对 co-run-con 元素进行分组元件。因此,您首先要为此

定义一个键
<xsl:key 
   name="con" 
   match="co-ru-con" 
   use="concat(generate-id(..), '|', si-ru-con/ru-con-ite/bId)" />

然后,无论何时匹配 o-ru-con 元素,您都会通过查找第一次出现的元素来匹配不同的 co-ru-con 密钥中的那个元素

<xsl:apply-templates 
   select="co-ru-con
      [generate-id() 
         = generate-id(key('con', concat(generate-id(..), '|', si-ru-con/ru-con-ite/bId))[1])]" />

这将允许您为该组创建单个 co-ru-con 元素。然后,您将匹配该组的子 si-ru-con 元素,如此

<xsl:apply-templates 
 select="key('con', concat(generate-id(..), '|', si-ru-con/ru-con-ite/bId))/si-ru-con" />

这是完整的XSLT

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
   <xsl:output method="xml" indent="yes"/>
   <xsl:strip-space elements="*" />
   <xsl:key name="con" match="co-ru-con" use="concat(generate-id(..), '|', si-ru-con/ru-con-ite/bId)" />

   <xsl:template match="o-ru-con">
      <o-ru-con>
         <xsl:apply-templates select="co-ru-con[generate-id() = generate-id(key('con', concat(generate-id(..), '|', si-ru-con/ru-con-ite/bId))[1])]" />
      </o-ru-con>
   </xsl:template>

   <xsl:template match="co-ru-con">
      <co-ru-con b-name="{@b-name}" bId="{si-ru-con/ru-con-ite/bId}">
         <xsl:apply-templates select="key('con', concat(generate-id(..), '|', si-ru-con/ru-con-ite/bId))/si-ru-con" />
      </co-ru-con>
   </xsl:template>

   <xsl:template match="bId" />

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

当应用于您的示例XML时,输出以下内容

<o-com-inter version="1" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
   <rules deplVer="125">
      <cre-det-r name="Auto_R_1">
         <act>false</act>
         <thres-curr>000</thres-curr>
         <thress>
            <thres name="Auto_R_125_1">
               <sco>11111</sco>
               <serv-res>
                  <serv-res>ABC</serv-res>
               </serv-res>
               <cat>
                  <cate>CatA</cate>
               </cat>
               <o-ru-con>
                  <co-ru-con b-name="InterMge" bId="2">
                     <si-ru-con>
                        <ru-con-ite>
                           <ru-con-ite-lit>
                              <nae1>ac</nae1>
                              <ope>T_EQ</ope>
                              <val1>000</val1>
                           </ru-con-ite-lit>
                        </ru-con-ite>
                     </si-ru-con>
                     <si-ru-con>
                        <ru-con-ite>
                           <ru-con-ite-lit>
                              <nae1>ac</nae1>
                              <ope>T_EQ</ope>
                              <val1>001</val1>
                           </ru-con-ite-lit>
                        </ru-con-ite>
                     </si-ru-con>
                  </co-ru-con>
                  <co-ru-con b-name="InterMge" bId="3">
                     <si-ru-con>
                        <ru-con-ite>
                           <ru-con-ite-lit>
                              <nae1>tra</nae1>
                              <ope>T_EQ</ope>
                              <val1>014</val1>
                           </ru-con-ite-lit>
                        </ru-con-ite>
                     </si-ru-con>
                     <si-ru-con>
                        <ru-con-ite>
                           <ru-con-ite-lit>
                              <nae1>tra</nae1>
                              <ope>T_EQ</ope>
                              <val1>015</val1>
                           </ru-con-ite-lit>
                        </ru-con-ite>
                     </si-ru-con>
                  </co-ru-con>
               </o-ru-con>
               <al>true</al>
            </thres>
         </thress>
         <description> rSId=125 Fp=0.1234567 Dr=1.0 rSId=48 SFp=1.0 SDr=1.0_2012-10-10T09:55:09+02:00 </description>
      </cre-det-r>
      <cre-det-r> ....If this block exists, it will contain the same elements as above in the same order, otherwise, this block doesn't exists and the code contains only the preceding "cre-det-r" block. </cre-det-r>
   </rules>
</o-com-inter>