我正在寻找基于标签的xml文件的简单拆分;比如3个标签(可选)总是重复并需要拆分,如下所示:
输入
<?xml version="1.0" encoding="UTF-8"?>
<Test>
<tag1>A</tag1>
<tag2>B</tag2>
<tag3>C</tag3>
<tag1>1</tag1>
<tag2>2</tag2>
<tag3>3</tag3>
<tag1>apple</tag1>
<tag2>orange</tag2>
<tag3>mango</tag3>
</Test>
预期产出
<Root>
<Test>
<tag1>A</tag1>
<tag2>B</tag2>
<tag3>C</tag3>
</Test>
<Test>
<tag1>1</tag1>
<tag2>2</tag2>
<tag3>3</tag3>
</Test>
<Test>
<tag1>apple</tag1>
<tag2>orange</tag2>
<tag3>mango</tag3>
</Test>
</Root>
这里的挑战是所有3个标签都是可选的,可以或不可以出现在一个块中。如果没有可选项 - 我的问题已在此处得到解答 - Split based on just tags
感谢任何帮助
由于
答案 0 :(得分:0)
如果我理解正确的要求,你想做:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:template match="/Test">
<Root>
<xsl:apply-templates/>
</Root>
</xsl:template>
<xsl:template match="tag1">
<Test>
<xsl:copy-of select=". | following-sibling::*[1][self::tag2 or self::tag3]"/>
<xsl:if test="following-sibling::*[1][self::tag2]">
<xsl:copy-of select="following-sibling::*[2][self::tag3] "/>
</xsl:if>
</Test>
</xsl:template>
<xsl:template match="tag2[not(preceding-sibling::*[1][self::tag1])]">
<Test>
<xsl:copy-of select=". | following-sibling::*[1][self::tag3]"/>
</Test>
</xsl:template>
<xsl:template match="tag3[not(preceding-sibling::*[1][self::tag2 or self::tag1])]">
<Test>
<xsl:copy-of select="."/>
</Test>
</xsl:template>
<xsl:template match="text()"/>
</xsl:stylesheet>
这将为每个开始一个新的组:
tag1
元素; tag2
元素,不会紧跟tag1
元素; tag3
元素,不会紧跟tag2
或tag1
元素。一个组以tag3
结尾,或者以序列中断前的最后一个元素结束,以先到者为准。
应用于以下测试输入:
<强> XML 强>
<Test>
<tag2>B</tag2>
<tag3>C</tag3>
<tag1>1</tag1>
<tag3>3</tag3>
<tag1>apple</tag1>
<tag2>orange</tag2>
<tag1>A</tag1>
<tag2>B</tag2>
<tag1>1</tag1>
<tag3>3</tag3>
<tag3>mango</tag3>
<tag1>A</tag1>
<tag2>B</tag2>
<tag3>C</tag3>
</Test>
结果将是:
<?xml version="1.0" encoding="UTF-8"?>
<Root>
<Test>
<tag2>B</tag2>
<tag3>C</tag3>
</Test>
<Test>
<tag1>1</tag1>
<tag3>3</tag3>
</Test>
<Test>
<tag1>apple</tag1>
<tag2>orange</tag2>
</Test>
<Test>
<tag1>A</tag1>
<tag2>B</tag2>
</Test>
<Test>
<tag1>1</tag1>
<tag3>3</tag3>
</Test>
<Test>
<tag3>mango</tag3>
</Test>
<Test>
<tag1>A</tag1>
<tag2>B</tag2>
<tag3>C</tag3>
</Test>
</Root>
答案 1 :(得分:0)
这是一种替代解决方案,可以处理任何大小的组,前提是该组中至少有一个元素始终存在。
在此示例中,每个组中最多包含5个元素,并且元素tag3
将出现在每个组中:
<强> XML 强>
<Test>
<tag1>A</tag1>
<tag2>B</tag2>
<tag3>C</tag3>
<tag4>D</tag4>
<tag5>E</tag5>
<tag2>2</tag2>
<tag3>3</tag3>
<tag5>5</tag5>
<tag3>C</tag3>
<tag3>3</tag3>
<tag4>4</tag4>
<tag5>5</tag5>
<tag2>B</tag2>
<tag3>C</tag3>
<tag4>D</tag4>
<tag1>1</tag1>
<tag3>3</tag3>
<tag5>5</tag5>
</Test>
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:key name="elems-before" match="tag1 | tag2" use="generate-id(following-sibling::tag3[1])" />
<xsl:key name="elems-after" match="tag4 | tag5" use="generate-id(preceding-sibling::tag3[1])" />
<xsl:template match="/Test">
<Root>
<xsl:for-each select="tag3">
<Test>
<xsl:copy-of select="key('elems-before', generate-id())"/>
<xsl:copy-of select="."/>
<xsl:copy-of select="key('elems-after', generate-id())"/>
</Test>
</xsl:for-each>
</Root>
</xsl:template>
</xsl:stylesheet>
<强>结果强>
<?xml version="1.0" encoding="UTF-8"?>
<Root>
<Test>
<tag1>A</tag1>
<tag2>B</tag2>
<tag3>C</tag3>
<tag4>D</tag4>
<tag5>E</tag5>
</Test>
<Test>
<tag2>2</tag2>
<tag3>3</tag3>
<tag5>5</tag5>
</Test>
<Test>
<tag3>C</tag3>
</Test>
<Test>
<tag3>3</tag3>
<tag4>4</tag4>
<tag5>5</tag5>
</Test>
<Test>
<tag2>B</tag2>
<tag3>C</tag3>
<tag4>D</tag4>
</Test>
<Test>
<tag1>1</tag1>
<tag3>3</tag3>
<tag5>5</tag5>
</Test>
</Root>