【问题标题】:How to move a node in XML using XSLT mapping with condition如何使用带条件的 XSLT 映射在 XML 中移动节点
【发布时间】: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 节点,它不会给出所需的结果。我知道我需要为每个周期使用两个,但不知道如何在这里实现它。

你能帮忙吗?

【问题讨论】:

    标签: xml xslt


    【解决方案1】:

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

    1. 第一个,对于根节点,将浏览每个E1MARCM 节点并建立一个与匹配的节点相等的节点。
    2. 第二个,对于每个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 并将其排除在外,但我可以稍后将其过滤掉。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2012-07-31
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2018-01-19
      相关资源
      最近更新 更多