如何使用带条件的 XSLT 映射在 XML 中移动节点

Posted

技术标签:

【中文标题】如何使用带条件的 XSLT 映射在 XML 中移动节点【英文标题】:How to move a node in XML using XSLT mapping with condition 【发布时间】:2019-01-28 04:29:39 【问题描述】:

我有一个类似这样的xml:

<?xml version="1.0"?>
<MATMAS05>
    <IDOC BEGIN="1">
        <EDI_DC40 SEGMENT="1">
            <CREDAT>20180822</CREDAT>
            <CRETIM>180201</CRETIM>
            <SERIAL>20180821134354</SERIAL>
        </EDI_DC40>
        <E1MARAM SEGMENT="1">
            <MSGFN>005</MSGFN>
            <MATNR>000000000002000010</MATNR>
            <E1MARA1 SEGMENT="1">
                <MAXC>0.000</MAXC>
                <MAXC_TOL>0.0</MAXC_TOL>
            </E1MARA1>
            <E1MAKTM SEGMENT="1">
                <MSGFN>005</MSGFN>
                <SPRAS>E</SPRAS>
            </E1MAKTM>
            <E1MARCM SEGMENT="1">
                <MSGFN>009</MSGFN>
                <WERKS>3030</WERKS>
                <E1MARDM SEGMENT="1">
                    <MSGFN>009</MSGFN>
                    <LGORT>1000</LGORT>
                </E1MARDM>
                <E1MPGDM SEGMENT="1">
                    <MSGFN>005</MSGFN>
                </E1MPGDM>
            </E1MARCM>
            <E1MARCM SEGMENT="1">
                <MSGFN>005</MSGFN>
                <WERKS>3040</WERKS>
                <E1MARDM SEGMENT="1">
                    <MSGFN>005</MSGFN>
                    <LGORT>1000</LGORT>
                </E1MARDM>
                <E1MPGDM SEGMENT="1">
                    <MSGFN>005</MSGFN>
                </E1MPGDM>
            </E1MARCM>
            <E1MARMM SEGMENT="1">
                <MSGFN>005</MSGFN>
                <MEINH>EA</MEINH>
            </E1MARMM>
            <E1MBEWM SEGMENT="1">
                <MSGFN>009</MSGFN>
                <BWKEY>3030</BWKEY>
            </E1MBEWM>
            <E1MBEWM SEGMENT="1">
                <MSGFN>005</MSGFN>
                <BWKEY>3040</BWKEY>
            </E1MBEWM>
            <E1MLANM SEGMENT="1">
                <MSGFN>005</MSGFN>
                <ALAND>AU</ALAND>
            </E1MLANM>
        </E1MARAM>
    </IDOC>
</MATMAS05>

我需要将 E1MBEWM 节点移动到适当的 E1MARCM 节点,其中 BWKEY=WERKS

所以,基本上应该是这样的输出:

    <?xml version="1.0"?>
<MATMAS05>
    <IDOC BEGIN="1">
        <EDI_DC40 SEGMENT="1">
            <CREDAT>20180822</CREDAT>
            <CRETIM>180201</CRETIM>
            <SERIAL>20180821134354</SERIAL>
        </EDI_DC40>
        <E1MARAM SEGMENT="1">
            <MSGFN>005</MSGFN>
            <MATNR>000000000002000010</MATNR>
            <E1MARA1 SEGMENT="1">
                <MAXC>0.000</MAXC>
                <MAXC_TOL>0.0</MAXC_TOL>
            </E1MARA1>
            <E1MAKTM SEGMENT="1">
                <MSGFN>005</MSGFN>
                <SPRAS>E</SPRAS>
            </E1MAKTM>
            <E1MARCM SEGMENT="1">
                <MSGFN>009</MSGFN>
                <WERKS>3030</WERKS>
                <E1MARDM SEGMENT="1">
                    <MSGFN>009</MSGFN>
                    <LGORT>1000</LGORT>
                </E1MARDM>
                <E1MPGDM SEGMENT="1">
                    <MSGFN>005</MSGFN>
                </E1MPGDM>
                <E1MBEWM SEGMENT="1">
                   <MSGFN>009</MSGFN>
                   <BWKEY>3030</BWKEY>
                </E1MBEWM>
            </E1MARCM>
            <E1MARCM SEGMENT="1">
                <MSGFN>005</MSGFN>
                <WERKS>3040</WERKS>
                <E1MARDM SEGMENT="1">
                    <MSGFN>005</MSGFN>
                    <LGORT>1000</LGORT>
                </E1MARDM>
                <E1MPGDM SEGMENT="1">
                    <MSGFN>005</MSGFN>
                </E1MPGDM>
                <E1MBEWM SEGMENT="1">
                   <MSGFN>005</MSGFN>
                   <BWKEY>3040</BWKEY>
                </E1MBEWM>
            </E1MARCM>
            <E1MARMM SEGMENT="1">
                <MSGFN>005</MSGFN>
                <MEINH>EA</MEINH>
            </E1MARMM>
            <E1MLANM SEGMENT="1">
                <MSGFN>005</MSGFN>
                <ALAND>AU</ALAND>
            </E1MLANM>
        </E1MARAM>
    </IDOC>
</MATMAS05>

可能的问题是有时 E1MBEWM 节点可能不在传入消息中。

我正在尝试使用类似的 XSLT:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="2.0">
 <xsl:output omit-xml-declaration="yes" indent="yes"/>
 <xsl:strip-space elements="*"/>
 <xsl:template match="node()|@*" name="identity">
  <xsl:copy>
   <xsl:apply-templates select="node()|@*"/>
  </xsl:copy>
 </xsl:template>
 <xsl:template match="E1MARCM/*[1]">
  <xsl:copy-of select="/*/E1MBEWM[1]"/>
  <xsl:call-template name="identity"/>
 </xsl:template>
 <xsl:template match="/*/E1MBEWM[1]"/>
</xsl:stylesheet>

但如果有多个 E1MARCM 节点,它不会给出所需的结果。我知道我需要为每个周期使用两个,但不知道如何在这里实现它。

你能帮忙吗?

【问题讨论】:

【参考方案1】:

我的建议是具有两个级别(两个模板)的 XSL:

    第一个,对于根节点,将浏览每个E1MARCM 节点并建立一个与匹配的节点相等的节点。

    第二个,对于每个E1MARCM 节点,将复制一个满足所需条件的节点E1MBEWM

    <?xml version="1.0" encoding="ISO-8859-1"?>
    <xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    
    <xsl:output omit-xml-declaration="yes" indent="yes"/>
    <xsl:strip-space elements="*"/>
    
    <xsl:template match="/XML">
    <XML>
        <xsl:apply-templates select="E1MARCM"/>
    </XML>
    </xsl:template>
    
    <xsl:template match="E1MARCM">
      <E1MARCM SEGMENT="@SEGMENT">
        <MSGFN><xsl:value-of select="MSGFN"/></MSGFN>
        <WERKS><xsl:value-of select="WERKS"/></WERKS>
        <xsl:variable name="mywerks" select="WERKS"></xsl:variable>
        <xsl:copy-of select="../E1MBEWM[BWKEY=$mywerks]"/>
      </E1MARCM>
    </xsl:template>
    
    </xsl:stylesheet>
    

(请注意,在将某个节点的值放入上下文节点不同的xpath表达式之前,必须将其存储在临时变量中。)

更新

而且,为了概括属性和子节点,还需要第三个目标:

<?xml version="1.0" encoding="ISO-8859-1"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:template match="/XML">
<XML>
    <xsl:apply-templates select="E1MARCM"/>
</XML>
</xsl:template>

<xsl:template match="E1MARCM">
  <E1MARCM>
    <xsl:apply-templates select="@*" mode="just-copy"/>
    <xsl:apply-templates select="*" mode="just-copy"/>
    <xsl:variable name="mywerks" select="WERKS"></xsl:variable>
    <xsl:copy-of select="../E1MBEWM[BWKEY=$mywerks]"/>
  </E1MARCM>
</xsl:template>

<xsl:template match="*|@*" mode="just-copy">
    <xsl:copy-of select="." />
</xsl:template>

</xsl:stylesheet>

【讨论】:

感谢您的建议。我不能真正列出 E1MARCM 段中的所有字段,因为它通常是这样的,它可能有/没有某些字段,具体取决于逻辑。所以,我肯定需要复制整个节点而不列出其中的字段。我实际上在根节点中有除 E1MARCM 之外的其他节点。我只是尝试简化示例以使其更易于理解。 看起来我已经过度简化了输入,这就是解决方案仍然需要调整的原因。我有以下(仍然是简化的)输入。这次我列出了我拥有的所有节点,每个节点都有几个字段。目的是获取所有节点,但将 E1MBEWM 节点放入适当的 E1MARCM 节点,其中 BWKEY=WERKS。我实际上可以在它们所在的地方拥有 E1MBEWM,但在 E1MARCM 中有适当的副本。我将在几分钟内将其添加为单独的回复 我刚刚用完整的结构更新了问题...现在它包含所有节点,我在每个节点中只留下了几个字段。【参考方案2】:

感谢小桑蒂。 我设法玩并找到了适合我目的的解决方案:

<?xml version="1.0" encoding="ISO-8859-1"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>

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

<xsl:template match="E1MARCM">
  <E1MARCM>
    <xsl:apply-templates select="@*" mode="just-copy"/>
    <xsl:apply-templates select="*" mode="just-copy"/>
    <xsl:variable name="mywerks" select="WERKS"></xsl:variable>
    <xsl:copy-of select="../E1MBEWM[BWKEY=$mywerks]"/>
  </E1MARCM>
</xsl:template>

<xsl:template match="*|@*" mode="just-copy">
    <xsl:copy-of select="." />
</xsl:template>

</xsl:stylesheet>

我有一些重复,因为现在我在 E1MARCM 中有 E1MBEWM 并将其排除在外,但我可以稍后将其过滤掉。

【讨论】:

以上是关于如何使用带条件的 XSLT 映射在 XML 中移动节点的主要内容,如果未能解决你的问题,请参考以下文章

当输入文件上的字段为空时,XSLT 2.0 如何跳过写入元素

如何使用xslt移动xml元素?

如何使用 XSLT 替换 XML 节点名称中的字符 - 更改根元素

使用带有条件的 XSLT 将 XML 转换为 XML

使用 XSLT 将 XML 元素移动到不同的节点

XSLT 1.0 / 使用条件对每个结果进行排序