XSLT muenchian 在子节点中按值分组

Posted

技术标签:

【中文标题】XSLT muenchian 在子节点中按值分组【英文标题】:XSLT muenchian grouping by value in child node 【发布时间】:2021-01-23 05:40:32 【问题描述】:

我正在尝试使用用 xslt 编写的 BizTalk 映射将 sap iDoc 转换为另一个 xml 文件。问题是我需要根据我们可以在特定子节点中找到的值对节点进行分组。

Muenchian 分组(我使用 XSLT 1.0)似乎是解决方案,但我不知道如何使其工作,因为我需要对子节点值进行分组。

<Receive>
    <idocData>
        <E2EDL20003GRP>
            <E2EDL2003>
                <ParentValue>PV</ParentValue>
            </E2EDL2003>
            ...
            <E2EDL24007GRP>
                <E2EDL24007>
                    <ChildValue>CHV1</ChildValue>
                </E2EDL24007>
                <E2EDL43000>
                    <QUALF>C</QUALF>
                    <BELNR>0000045690</BELNR>
                </E2EDL43000>
                ...
            </E2EDL24007GRP>
            <E2EDL24007GRP>
                <E2EDL24007>
                    <ChildValue>CHV2</ChildValue>
                </E2EDL24007>
                <E2EDL43000>
                    <QUALF>C</QUALF>
                    <BELNR>0000045690</BELNR>
                </E2EDL43000>
                ...
            </E2EDL24007GRP>
            <E2EDL24007GRP>
                <E2EDL24007>
                    <ChildValue>CHV3</ChildValue>
                </E2EDL24007>
                <E2EDL43000>
                    <QUALF>C</QUALF>
                    <BELNR>0000045691</BELNR>
                </E2EDL43000>
                ...
            </E2EDL24007GRP>
        </E2EDL20003GRP>
    </idocData>
</Receive>

应用 XSLT 转换后,我正在寻找的是:

<ns0:Root>
    <RecordA>
        <ID>PV</ID>
        <RecordB>
            <ID>0000045690</ID>
            <RecordC>
                <Value>CHV1</Value>
            </RecordC>
            <RecordC>
                <Value>CHV2</Value>
            </RecordC>
        </RecordB>
        <RecordB>
            <ID>0000045691</ID>
            <RecordC>
                <Value>CHV3</Value>
            </RecordC>
        </RecordB>
    </RecordA>
</ns0:Root>

如您所见,我需要将 E2EDL24007GRP 按E2EDL43000[QUALF='C']/BELNR 分组。 我尝试了以下 muenchian 分组:

<xsl:key name="command" match="s0:E2EDL24007GRP" use="s0:E2EDL43000[s0:QUALF='C']/s0:BELNR" />

<xsl:template match="/">
    <xsl:apply-templates select="/s1:Receive" />
</xsl:template>

<xsl:template match="/s1:Receive/s1:idocData">
    <xsl:for-each select="s0:E2EDL20003GRP">
        <ns0:Root>
            <!-- Record A -->
            <RecordA>
                <ID>
                    <xsl:value-of select="s0:E2EDL2003/s0:ParentValue" />
                </ID>
                <xsl:apply-templates select="s0:E2EDL24007GRP[generate-id()=generate-id(key('command',s0:E2EDL43000[s0:QUALF='C']/s0:BELNR)[1])]"/>
            </RecordA>
        </ns0:Root>
    </xsl:for-each>
</xsl:template>

<xsl:template match="E2EDL24007GRP">
...
</xsl:template>

但它根本不起作用,有什么想法吗?

【问题讨论】:

E2EDL20003GRP 可以有多个吗?如果是,两个E2EDL24007GRP 元素是否可以具有相同的BELNR 值但属于不同的E2EDL20003GRP 父母? 否,如果有多个 E2EDL20003GRP(这将非常罕见),则 BELNR 值将不同。它们只能在同一个 E2EDL20003GR 中相同。 【参考方案1】:

以下样式表:

XSLT 1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:ns0="http://example.com">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes" />

<xsl:key name="k" match="E2EDL24007GRP" use="E2EDL43000/BELNR" />

<xsl:template match="/Receive">
    <ns0:Root>
        <xsl:for-each select="idocData/E2EDL20003GRP">
            <RecordA>
                <ID>
                    <xsl:value-of select="E2EDL2003/ParentValue" />
                </ID>
                <xsl:for-each select="E2EDL24007GRP[generate-id()=generate-id(key('k', E2EDL43000/BELNR)[1])]">
                    <RecordB>
                        <ID>
                            <xsl:value-of select="E2EDL43000/BELNR" />
                        </ID>
                        <xsl:for-each select="key('k', E2EDL43000/BELNR)">
                            <RecordC>
                                <Value>
                                    <xsl:value-of select="E2EDL24007/ChildValue" />
                                </Value>
                            </RecordC>
                         </xsl:for-each>
                    </RecordB>
                </xsl:for-each>
            </RecordA>
        </xsl:for-each>
    </ns0:Root>
</xsl:template>

</xsl:stylesheet>

应用于您的输入示例,将产生:

结果

<?xml version="1.0" encoding="UTF-8"?>
<ns0:Root xmlns:ns0="http://example.com">
  <RecordA>
    <ID>PV</ID>
    <RecordB>
      <ID>0000045690</ID>
      <RecordC>
        <Value>CHV1</Value>
      </RecordC>
      <RecordC>
        <Value>CHV2</Value>
      </RecordC>
    </RecordB>
    <RecordB>
      <ID>0000045691</ID>
      <RecordC>
        <Value>CHV3</Value>
      </RecordC>
    </RecordB>
  </RecordA>
</ns0:Root>

注意

    您的输入示例没有命名空间; 您的输出具有未绑定到命名空间的前缀,这是不允许的;我使用了一个虚假的命名空间来生成格式良好的 XML 文档。

【讨论】:

以上是关于XSLT muenchian 在子节点中按值分组的主要内容,如果未能解决你的问题,请参考以下文章

XSLT / Muenchian 分组:如何从组中选择具有某些子元素的元素?

XSLT1,Muenchian 分组,在 foreach 循环中列出分组节点

Muenchian Grouping - 在一个节点内分组,而不是在整个文档内

XSLT 和 Muenchian 分组, 多级样本

Muenchian 分组 XSLT 1.0 多重分组

使用 XSLT 对 HTML 输出进行分组(muenchian 分组?)