使用Muenchian方法对XSLT1.0进行分组




我有以下XML

 `<?xml version="1.0" encoding="UTF-8"?>
<resultSet>
    <row>
        <ACCT_NO>79501</ACCT_NO>
        <PVT_CHOICE_CD>C1</PVT_CHOICE_CD>
        <PVT_STATUS>O</PVT_STATUS>
        <OVEERRIDE_STATUS />
        <EFFCTV_DATE>2009-11-24 01:58:06.000001</EFFCTV_DATE>
        <PREF_PROVIDER />
        <DEST_EML_ADDR>TEST1.TEST1@XYZ.COM</DEST_EML_ADDR>                                                                                                                                                                                                                                          
        <EML_PVT_TYPE_CD>O</EML_PVT_TYPE_CD>
        <REP_ID />
        <OVERD_RSN_CD />
        <OVERD_DESC />
        <OVERD_EFF_TS />
    </row>
    <row>
        <ACCT_NO>79501</ACCT_NO>
        <PVT_CHOICE_CD>D1</PVT_CHOICE_CD>
        <PVT_STATUS>O</PVT_STATUS>
        <OVEERRIDE_STATUS />
        <EFFCTV_DATE>2013-11-24 01:58:06.000001</EFFCTV_DATE>
        <PREF_PROVIDER />
        <DEST_EML_ADDR>TEST1.TEST1@XYZ.COM</DEST_EML_ADDR>                                                                                                                                                                                                                                          
        <EML_PVT_TYPE_CD>O</EML_PVT_TYPE_CD>
        <REP_ID />
        <OVERD_RSN_CD />
        <OVERD_DESC />
        <OVERD_EFF_TS />
    </row>
    <row>
        <ACCT_NO>79500</ACCT_NO>
        <PVT_CHOICE_CD>D1</PVT_CHOICE_CD>
        <PVT_STATUS>O</PVT_STATUS>
        <OVEERRIDE_STATUS />
        <EFFCTV_DATE>2012-12-23 00:12:23.000001</EFFCTV_DATE>
        <PREF_PROVIDER />
        <DEST_EML_ADDR>TEST2.TEST2@XYZ.COM </DEST_EML_ADDR>                                                                                                                                                                                                                                           
        <EML_PVT_TYPE_CD>O</EML_PVT_TYPE_CD>
        <REP_ID />
        <OVERD_RSN_CD />
        <OVERD_DESC />
        <OVERD_EFF_TS />
    </row>
</resultSet>`

此resultSet来自数据库调用,它表示卡帐户及其首选代码之间的多对多关系。卡可能与电子邮件(DEST_EML_ADDR(和电子邮件选项(EML_PVT_TYPE_CD(相关联,并且这两个标签对于多行中的同一卡号也将具有相同的值。卡片还将有一个唯一的偏好代码和一组与偏好相关的信息,这些信息对每一行都是唯一的

我的目标是提取每个唯一的卡号dest_email_addr和eml_pvt_type_cd,并将所有与偏好相关的信息关联到每个帐户的一个区块中。总之,上述输入的期望输出为:

    <?xml version="1.0" encoding="UTF-8"?>
<EntityResponse>
    <EntityDetails>
        <ACCT_NO>79501 </ACCT_NO>
        <DEST_EML_ADDR>TEST1.TEST1@XYZ.COM </DEST_EML_ADDR>
        <EML_PVT_TYPE_CD>O</EML_PVT_TYPE_CD>
        <PreferenceGroup>
                    <PreferenceDetails>
                                   <PVT_CHOICE_CD>C1</PVT_CHOICE_CD>
                                    <PVT_STATUS>O</PVT_STATUS>
                                    <OVEERRIDE_STATUS />
                                    <EFFCTV_DATE>2009-11-2401:58:06.000001</EFFCTV_DATE> 
                                    <PREF_PROVIDER />
                                    <REP_ID />
                                    <OVERD_RSN_CD />
                                    <OVERD_DESC />
                                    <OVERD_EFF_TS />
                    </PreferenceDetails>
                    <PreferenceDetails>
                                      <PVT_CHOICE_CD>D1</PVT_CHOICE_CD>
                                        <PVT_STATUS>O</PVT_STATUS>
                                        <OVEERRIDE_STATUS />
                                        <EFFCTV_DATE>2013-11-24 01:58:06.000001</EFFCTV_DATE>
                                         <PREF_PROVIDER />
                                         <REP_ID />
                                        <OVERD_RSN_CD />
                                        <OVERD_DESC />
                                        <OVERD_EFF_TS />
                    </PreferenceDetails>
            </PreferenceGroup>
       </EntityDetails>
       <EntityDetails>
                       <ACCT_NO>79500</ACCT_NO>
                       <DEST_EML_ADDR>TEST2.TEST2@XYZ.COM</DEST_EML_ADDR>
                        <EML_PVT_TYPE_CD>O</EML_PVT_TYPE_CD>
                        <PreferenceGroup>
                             <PreferenceDetails>
                                <PVT_CHOICE_CD>D1</PVT_CHOICE_CD>
                                <PVT_STATUS>O</PVT_STATUS>
                                <OVEERRIDE_STATUS />
                                <EFFCTV_DATE>2012-12-23 00:12:23.000001</EFFCTV_DATE>
                                <PREF_PROVIDER />
                                <REP_ID />
                                <OVERD_RSN_CD />
                                <OVERD_DESC />
                                <OVERD_EFF_TS />
                          </PreferenceDetails>
                        </PreferenceGroup>
       </EntityDetails>

    </EntityResponse>

我被限制使用XSLT1.0,在阅读了这个基于多键线程的Muenchian方法分组后,我意识到这也是我需要的解决方案。基于这些信息,我创建了以下XSL,其中我使用ACCT_NO、DEST_EML_ADDR和EML_PVT_TYPE_CD创建了一个密钥,以过滤掉唯一元素,并基于第二个密钥应用第二个模板来进一步分组一对多偏好代码关系:

    <?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
   <xsl:output method="xml" indent="yes"/>
   <xsl:key name="entity-email-key" match="row" use="concat(ACCT_NO,'~#~',DEST_EML_ADDR, '~#~',EML_PVT_TYPE_CD)"/>
   <xsl:key name="prefcd-key" match="row" use="concat(ACCT_NO,'~#~',DEST_EML_ADDR, '~#~',EML_PVT_TYPE_CD, '~#~',PVT_CHOICE_CD)"/>

   <xsl:template match="/resultSet">
      <EntityResponse>
                  <xsl:apply-templates select="row[generate-id() = generate-id(key('entity-email-key', concat(ACCT_NO,'~#~',DEST_EML_ADDR, '~#~',EML_PVT_TYPE_CD))[1])]" mode="entity-email-key"/>
      </EntityResponse>
   </xsl:template>
   <xsl:template match="row" mode="entity-email-key">
            <EntityDetails>
                   <xsl:copy-of select="ACCT_NO|PVT_STATUS|DEST_EML_ADDR|EML_PVT_TYPE_CD"/>
                       <PreferenceGroup>
                        <xsl:apply-templates select="key('entity-email-key', concat(ACCT_NO,'~#~',DEST_EML_ADDR, '~#~',EML_PVT_TYPE_CD))[generate-id() = generate-id(key('prefcd-key', concat(ACCT_NO,'~#~',DEST_EML_ADDR, '~#~',EML_PVT_TYPE_CD,PVT_CHOICE_CD))[1])]" mode="prefcd-key"/>
                      </PreferenceGroup>
           </EntityDetails>

   </xsl:template>

   <xsl:template match="row" mode="prefcd-key">
         <PreferenceDetails>
                          <xsl:copy-of select="PVT_CHOICE_CD|PVT_STATUS|OVEERRIDE_STATUS|EFFCTV_DATE|PREF_PROVIDER|REP_ID|OVERD_RSN_CD|OVERD_DESC|OVERD_EFF_TS"/>
         </PreferenceDetails>
   </xsl:template>

</xsl:stylesheet>

但当我运行代码时,我意识到它只按ACCT_NO、EMAIL和EML_PVT_TYPE_CD进行分组,而不能按与首选项代码相关的标签进行分组。我尝试了选择偏好代码数据的不同组合,我能得到的最接近的是每张卡都有一个偏好代码,但偏好代码可能不止一个,就像第一张卡的情况一样,它有两个唯一的偏好代码相关数据。

因此,我认为我对这种方法还没有很好的理解,也许我甚至不需要定义第二个关键,但在与这个问题斗争了几天后,是时候向社区求助了,看看我缺少了什么
非常感谢。

查看问题中的XSLT,在使用prefcd-key键时,您遗漏了concat语句中最后一个PVT_CHOICE_CD之前的分隔符~#~

试试这个:

 <xsl:apply-templates select="key('entity-email-key', concat(ACCT_NO,'~#~',DEST_EML_ADDR, '~#~',EML_PVT_TYPE_CD))
     [generate-id() = generate-id(key('prefcd-key', concat(ACCT_NO,'~#~',DEST_EML_ADDR, '~#~',EML_PVT_TYPE_CD, '~#~', PVT_CHOICE_CD))[1])]" 
     mode="prefcd-key"/>

最新更新