如何使用带条件的 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 如何跳过写入元素