使用这个小的XSL变换,我可以给出一个"平坦" XML结构稍微层次结构:
<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:key name="child1" match="item[meta/para/level]" use="generate-id(preceding-sibling::item[meta/para/level < current()/meta/para/level][1])"/>
<xsl:key name="child2" match="item[meta/text]" use="generate-id(preceding-sibling::item[meta/para/level][1])"/>
<xsl:template match="/doc">
<xsl:variable name="min"> <!-- xslt2.0 min() -->
<xsl:for-each select="item/meta/para/level">
<xsl:sort select="." data-type="number"/>
<xsl:if test="position() = 1">
<xsl:value-of select="."/>
</xsl:if>
</xsl:for-each>
</xsl:variable>
<root>
<xsl:apply-templates select="item[meta/para/level = $min] | key('child2', '')"/>
</root>
</xsl:template>
<xsl:template match="item[meta/para/level]">
<test title="{meta/para/title}">
<xsl:apply-templates select="key('child1', generate-id()) | key('child2', generate-id())"/>
</test>
</xsl:template>
<xsl:template match="item[meta/text]">
<text>
<xsl:value-of select="meta/text"/>
</text>
</xsl:template>
</xsl:stylesheet>
使用平面结构应用于此XML示例
<?xml version="1.0" encoding="UTF-8"?>
<doc>
<item>
<meta>
<text>abc</text>
</meta>
</item>
<item>
<meta>
<para>
<level>1</level>
<title>a</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>2</level>
<title>b</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>3</level>
<title>c</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>4</level>
<title>d</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>4</level>
<title>e</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>5</level>
<title>f</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>2</level>
<title>g</title>
</para>
</meta>
</item>
<item>
<meta>
<text>def</text>
</meta>
</item>
<item>
<meta>
<text>ghi</text>
</meta>
</item>
<item>
<meta>
<para>
<level>4</level>
<title>h</title>
</para>
</meta>
</item>
</doc>
是(所需的)层次结果,如下:
<?xml version="1.0" encoding="UTF-8"?>
<root>
<text>abc</text>
<test title="a">
<test title="b">
<test title="c">
<test title="d"/>
<test title="e">
<test title="f"/>
</test>
</test>
</test>
<test title="g">
<text>def</text>
<text>ghi</text>
<test title="h"/>
</test>
</test>
</root>
如果"级别"信息不是由整数确定的,而是按字符串的长度确定,则转换不会成功。
<?xml version="1.0" encoding="UTF-8"?>
<doc>
<item>
<meta>
<text>abc</text>
</meta>
</item>
<item>
<meta>
<para>
<level>a</level>
<title>a</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>bc</level>
<title>b</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>def</level>
<title>c</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>ghij</level>
<title>d</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>klmn</level>
<title>e</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>opqrs</level>
<title>f</title>
</para>
</meta>
</item>
<item>
<meta>
<para>
<level>tu</level>
<title>g</title>
</para>
</meta>
</item>
<item>
<meta>
<text>def</text>
</meta>
</item>
<item>
<meta>
<text>ghi</text>
</meta>
</item>
<item>
<meta>
<para>
<level>vwxy</level>
<title>h</title>
</para>
</meta>
</item>
</doc>
通过调用"字符串长度"来查找最小值的功能:
<xsl:sort select="string-length(.)" data-type="number"/>
应该如何使用键,以便将字符串的长度评估为级别信息?
在比较 meta/para/level
的密钥 child1
中,您需要计算情况的 string-length
其中<level>
具有字符串值而不是数字。如下:
<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:key name="child1" match="item[meta/para/level]"
use="generate-id(preceding-sibling::item[string-length(meta/para/level) < string-length(current()/meta/para/level)][1])"/>
<xsl:key name="child2" match="item[meta/text]"
use="generate-id(preceding-sibling::item[meta/para/level][1])"/>
<xsl:template match="/doc">
<xsl:variable name="min"> <!-- xslt2.0 min() -->
<xsl:for-each select="item/meta/para/level">
<xsl:sort select="string-length(.)" data-type="number"/>
<xsl:if test="position() = 1">
<xsl:value-of select="."/>
</xsl:if>
</xsl:for-each>
</xsl:variable>
<root>
<xsl:apply-templates select="item[meta/para/level = $min] | key('child2', '')"/>
</root>
</xsl:template>
<xsl:template match="item[meta/para/level]">
<test title="{meta/para/title}">
<xsl:apply-templates select="key('child1', generate-id()) | key('child2', generate-id())"/>
</test>
</xsl:template>
<xsl:template match="item[meta/text]">
<text>
<xsl:value-of select="meta/text"/>
</text>
</xsl:template>
</xsl:stylesheet>
请在此处参考
@vebbie几乎正确(谢谢!)。生成密钥时,必须排除非合适的节点很重要:<xsl:key name="child1" match="item[meta/para]"
use="generate-id(preceding-sibling::item[meta/para and (string-length(meta/para/level) < string-length(current()/meta/para/level))][1])"/>
,因此完整的答案为
<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:key name="child1" match="item[meta/para]"
use="generate-id(preceding-sibling::item[meta/para and (string-length(meta/para/level) < string-length(current()/meta/para/level))][1])"/>
<xsl:key name="child2" match="item[meta/text]"
use="generate-id(preceding-sibling::item[meta/para][1])"/>
<xsl:template match="/doc">
<xsl:variable name="min">
<!-- xslt2.0 min() -->
<xsl:for-each select="item/meta/para/level">
<xsl:sort select="string-length(.)" data-type="number"/>
<xsl:if test="position() = 1">
<xsl:value-of select="string-length(.)"/>
</xsl:if>
</xsl:for-each>
</xsl:variable>
<root>
<xsl:apply-templates select="item[string-length(meta/para/level) = $min] | key('child2', '')"/>
</root>
</xsl:template>
<xsl:template match="item[meta/para]">
<test title="{meta/para/title}">
<xsl:apply-templates select="key('child1', generate-id()) | key('child2', generate-id())"/>
</test>
</xsl:template>
<xsl:template match="item[meta/text]">
<text>
<xsl:value-of select="meta/text"/>
</text>
</xsl:template>
</xsl:stylesheet>