平面 XML 到层次结构 XML



我正在研究称为webmethods(集成工具(的中间件技术,我们有一个问题陈述,需要帮助/指导。

我们得到了一个平面XML,如下所示,XML需要转换为层次结构结构。XML 包含ComponentIDrelatedComponentID其中如果RelatedCompID为空或null,那就是我的父级,如果RelatedCompID不是null那就是我的孩子。

所以基本上它是一个深度递归,它可以包含 n 个父项和n个子项。如下所示;

ComponentID=0001 is the parent
ComponentID=0002 is the parent
ComponentID=003 is the child of parent 0001

。等等。它可以有任何孩子和任何父母。

现在,我们需要将其转换为分层结构

我们尝试使用网络方法流语言,但没有奏效。我们也尝试用java代码编写,但没有用。最后,我们想尝试使用无法生成正确逻辑的 XSLT

<?xml version="1.0" encoding="UTF-8"?>
<LineItems>
<LineItem>
<ComponentID>0001</ComponentID>
<RelatedCompID></RelatedCompID>
<Category>MainComponent</Category>         
</LineItem>
<LineItem>
<ComponentID>0002</ComponentID>
<RelatedCompID></RelatedCompID>
<Category>MainComponent</Category>         
</LineItem>
<LineItem>
<ComponentID>0003</ComponentID>
<RelatedCompID></RelatedCompID>
<Category>MainComponent</Category>         
</LineItem>
<LineItem>
<ComponentID>003</ComponentID>
<RelatedCompID>0001</RelatedCompID>
<Category>SubComponent</Category>            
</LineItem>
<LineItem>
<ComponentID>004</ComponentID>
<RelatedCompID>0001</RelatedCompID>
<Category>SubComponent</Category>
</LineItem>
<LineItem>
<ComponentID>055</ComponentID>
<RelatedCompID>003</RelatedCompID>
<Category>SubComponent</Category>
</LineItem>
<LineItem>
<ComponentID>066</ComponentID>
<RelatedCompID>003</RelatedCompID>
<Category>SubComponent</Category>
</LineItem>
<LineItem>
<ComponentID>777</ComponentID>
<RelatedCompID>055</RelatedCompID>
<Category>SubComponent</Category>
</LineItem>
<LineItem>
<ComponentID>008</ComponentID>
<RelatedCompID>0002</RelatedCompID>
<Category>SubComponent</Category>
</LineItem>
<LineItem>
<ComponentID>099</ComponentID>
<RelatedCompID>004</RelatedCompID>
<Category>SubComponent</Category>
</LineItem>
<LineItem>
<ComponentID>123</ComponentID>
<RelatedCompID>0002</RelatedCompID>
<Category>SubComponent</Category>
</LineItem>
</LineItems>

使用XSLT,它似乎是对键的简单使用:

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
exclude-result-prefixes="#all"
version="3.0">
<xsl:output indent="yes"/>
<xsl:key name="child" match="LineItem" use="RelatedCompID"/>
<xsl:template match="/*">
<Components>
<xsl:apply-templates select="key('child', '')"/>
</Components>
</xsl:template>
<xsl:template match="LineItem">
<Component>
<xsl:copy-of select="* except RelatedCompID"/>
<SubComponents>
<xsl:apply-templates select="key('child', ComponentID)"/>
</SubComponents>
</Component>
</xsl:template>
</xsl:stylesheet>

https://xsltfiddle.liberty-development.net/gWvjQgn

这会将您的输入转换为

<Components>
<Component>
<ComponentID>0001</ComponentID>
<Category>MainComponent</Category>
<SubComponents>
<Component>
<ComponentID>003</ComponentID>
<Category>SubComponent</Category>
<SubComponents>
<Component>
<ComponentID>055</ComponentID>
<Category>SubComponent</Category>
<SubComponents>
<Component>
<ComponentID>777</ComponentID>
<Category>SubComponent</Category>
<SubComponents/>
</Component>
</SubComponents>
</Component>
<Component>
<ComponentID>066</ComponentID>
<Category>SubComponent</Category>
<SubComponents/>
</Component>
</SubComponents>
</Component>
<Component>
<ComponentID>004</ComponentID>
<Category>SubComponent</Category>
<SubComponents>
<Component>
<ComponentID>099</ComponentID>
<Category>SubComponent</Category>
<SubComponents/>
</Component>
</SubComponents>
</Component>
</SubComponents>
</Component>
<Component>
<ComponentID>0002</ComponentID>
<Category>MainComponent</Category>
<SubComponents>
<Component>
<ComponentID>008</ComponentID>
<Category>SubComponent</Category>
<SubComponents/>
</Component>
<Component>
<ComponentID>123</ComponentID>
<Category>SubComponent</Category>
<SubComponents/>
</Component>
</SubComponents>
</Component>
<Component>
<ComponentID>0003</ComponentID>
<Category>MainComponent</Category>
<SubComponents/>
</Component>
</Components>

显然,我必须为元素名称做出一些选择(例如Component(和结构(例如SubComponents(,但您应该能够根据问题中未显示的需求进行调整。

正如您抱怨由于使用except而导致的错误,这是任何 XSLT 2 处理器都应该支持的 XPath 2 运算符,但是使用谓词*[not(self::RelatedCompID)]为 XSLT/XPath 1 重写该表达式很容易,因此 XSLT 1 版本将是

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
version="1.0">
<xsl:output indent="yes"/>
<xsl:key name="child" match="LineItem" use="RelatedCompID"/>
<xsl:template match="/*">
<Components>
<xsl:apply-templates select="key('child', '')"/>
</Components>
</xsl:template>
<xsl:template match="LineItem">
<Component>
<xsl:copy-of select="*[not(self::RelatedCompID)]"/>
<SubComponents>
<xsl:apply-templates select="key('child', ComponentID)"/>
</SubComponents>
</Component>
</xsl:template>
</xsl:stylesheet>

在线 https://xsltfiddle.liberty-development.net/gWvjQgn/3。

至于获得确切 XML 输出结构的帮助,如果您无法弄清楚如何调整此答案中的代码示例以产生所需的输出,那么请编辑您的问题并将其显示为 XML 代码示例。

最新更新