XSLT 1.0: мюнхенская группировка не работает

У меня есть следующая упрощенная структура XML:

<?xml version="1.0" encoding="utf-8"?>
<list>
<INVOIC>
    <M_INVOIC>
        <G_SG25>
            <S_LIN>
                <id>LIN</id>
                <D_1082>1</D_1082>
                <C_C212>
                    <D_7140>7610400271943</D_7140>
                    <D_7143_3>EN</D_7143_3>
                </C_C212>
            </S_LIN>
        </G_SG25>
        <G_SG25>
            <S_LIN>
                <id>LIN</id>
                <D_1082>2</D_1082>
                <C_C212>
                    <D_7140>1234567890123</D_7140>
                    <D_7143_3>EN</D_7143_3>
                </C_C212>
            </S_LIN>
        </G_SG25>
    </M_INVOIC>
</INVOIC>
<INVOIC>
    <SALESORDER>
        <ET_VBAP>
            <item>
                <VBELN>0010002695</VBELN>
                <POSNR>000010</POSNR>
                <MATNR>000000000000400487</MATNR>
                <EAN11>1234567890123</EAN11>
            </item>
            <item>
                <VBELN>0010002695</VBELN>
                <POSNR>000020</POSNR>
                <MATNR>000000000000002054</MATNR>
                <EAN11>5012454920549</EAN11>
            </item>
            <item>
                <VBELN>0010002695</VBELN>
                <POSNR>000030</POSNR>
                <MATNR>000000000000392104</MATNR>
                <EAN11>3046920921046</EAN11>
            </item>
            <item>
                <VBELN>0010002695</VBELN>
                <POSNR>000040</POSNR>
                <MATNR>000000000000859146</MATNR>
                <EAN11>8003340591469</EAN11>
            </item>
            <item>
                <VBELN>0010002695</VBELN>
                <POSNR>000050</POSNR>
                <MATNR>000000000000727194</MATNR>
                <EAN11>7610400271943</EAN11>
            </item>
        </ET_VBAP>
    </SALESORDER>
</INVOIC>
</list>

мой XSLT:

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

<xsl:key name="kByEanPos" match="G_SG25" use="S_LIN/C_C212/D_7140"/>

<xsl:template match="/">

    <xsl:variable name="uniqueSet" select="G_SG25[generate-id()=generate-id(key('kByEanPos',S_LIN/C_C212/D_7140))]"/>
    <list>
        <xsl:for-each select="list/INVOIC/M_INVOIC/G_SG25[generate-id()=
            generate-id(key('kByEanPos',S_LIN/C_C212/D_7140))]">
            <ean>
                <xsl:value-of select="parent::M_INVOIC/parent::INVOIC/parent::list/INVOIC/SALESORDER/ET_VBAP/item/MATNR"/>
            </ean>
        </xsl:for-each>
    </list>
</xsl:template>
</xsl:stylesheet>

дает мне этот вывод XML:

<?xml version="1.0" encoding="UTF-8"?>
<list>
<ean>000000000000400487</ean>
<ean>000000000000400487</ean>
</list>

Но мой ожидаемый вывод XML:

<?xml version="1.0" encoding="UTF-8"?>
<list>
<ean>000000000000727194</ean>
<ean>000000000000400487</ean>
</list>

Я не уверен, что я делаю неправильно, я не могу найти свою ошибку. Я думаю, что это связано с ключом, который я определил. По сути, мне нужен ключ на <D_7140>, а затем найдите этот номер в структуре ниже в EAN11 и выведите MATNR прямо перед этим.

Спасибо за любые идеи и с наилучшими пожеланиями, Петр


person Peter    schedule 23.01.2013    source источник


Ответы (2)


Если вы пытаетесь найти элементы item на основе их значения EAN11, возможно, стоит подумать об использовании ключа для этого.

<xsl:key name="item" match="item" use="EAN11" />

Таким образом, вы можете уменьшить xsl:value-of до этого

<xsl:value-of select="key('item', S_LIN/C_C212/D_7140)/MATNR"/>
person Tim C    schedule 23.01.2013
comment
Привет Тим, Спасибо. Ваше решение работает отлично. Я адаптировал свое сопоставление, используя элемент сопоставления ключей и используя EAN11. С уважением, Питер - person Peter; 24.01.2013

В этой строке:

<xsl:value-of 
  select="parent::M_INVOIC/parent::INVOIC/parent::list/INVOIC/SALESORDER/ET_VBAP/item/MATNR"/>

Вы карабкаетесь вверх по дереву узлов и возвращаетесь к MATNR, поэтому единственное, что он когда-либо найдет, — это первый MATNR в документе. Чтобы найти MATNR, который соответствует текущему D_7140 в вашем for-each, это должно работать:

<xsl:value-of 
  select="/list/INVOIC/SALESORDER/ET_VBAP/item[EAN11 = current()/S_LIN/C_C212/D_7140]/MATNR"/>
person JLRishe    schedule 23.01.2013
comment
Привет JLRishe, Спасибо за ответ - работает (+1). Я предпочитаю решение Tim C, потому что оно позволяет избежать длинного XPATH. С уважением, Питер - person Peter; 24.01.2013