XSLT通过递归模板、条件匹配和上下文定位将扁平XML构建成嵌套结构:可依level属性、parent-id关联或缩进/序号推断层级,推荐用key索引提升效率,避免多层for-each嵌套。

XSLT 可以通过识别层级标识(比如 level、parent-id、缩进空格、序号前缀等)把扁平 XML 数据构建成嵌套的父子结构。核心思路是:用 递归模板 + 条件匹配 + 上下文定位,而不是靠循环。
用 level 字段构建层级
如果源数据每个节点带 level 属性(如 1 表示根,2 表示子,3 表示孙),最直观的方式是用 xsl:for-each 配合 xsl:if 找出当前层级的所有节点,再递归处理其子节点:
<xsl:template name="build-tree">
<xsl:param name="nodes" />
<xsl:param name="current-level" select="1" />
<p><xsl:for-each select="$nodes[level = $current-level]">
<item>
<xsl:attribute name="id"><xsl:value-of select="id"/></xsl:attribute>
<name><xsl:value-of select="name"/></name></p><pre class='brush:php;toolbar:false;'> <!-- 递归处理下一级子节点 -->
<xsl:variable name="children"
select="$nodes[level = $current-level + 1 and parent-id = current()/id]" />
<xsl:if test="$children">
<children>
<xsl:call-template name="build-tree">
<xsl:with-param name="nodes" select="$children" />
<xsl:with-param name="current-level" select="$current-level + 1" />
</xsl:call-template>
</children>
</xsl:if>
</item></xsl:for-each> </xsl:template>
调用时传入全部节点和起始 level 即可:<xsl:call-template name="build-tree"><xsl:with-param name="nodes" select="/data/item"/></xsl:call-template>
用 parent-id 关联父子关系
若数据含 parent-id(根节点 parent-id 为空或为 0),适合用 键(key)预索引 提升效率:
- 先定义 key:
<xsl:key name="children" match="item" use="parent-id"/> - 对根节点(
not(parent-id) or parent-id = '' or parent-id = '0')应用模板 - 在模板中用
key('children', current()/id)获取所有直接子节点,再递归
这样避免每次遍历全量数据,XSLT 1.0 也能高效处理几百上千条记录。
从缩进文本或序号推断层级(无显式字段)
当输入是类似 Markdown 或日志格式的扁平文本(如 1. 一级、1.1 二级、 二级(两个全角空格)),可用字符串函数提取层级线索:
-
normalize-space()去首尾空格,再用string-length() - string-length(normalize-space())算缩进空格数 - 用
substring-before(., ' ')提取序号前缀,再用translate()去掉点号,统计层级深度 - 把计算出的 “level” 存为临时变量,后续逻辑就和第一种方式一致
注意:XSLT 1.0 不支持正则,复杂模式建议先用外部工具预处理;XSLT 2.0+ 可用 analyze-string 或 tokenize() 更稳妥。
避免常见坑
- 别用
xsl:for-each嵌套多层模拟树——性能差且难维护 - 递归必须有明确终止条件(如无子节点时跳过
xsl:call-template) - 用
current()而非.在谓词里,防止上下文错位 - 测试时用小样本,关注生成的嵌套深度是否符合预期
基本上就这些。关键是选对层级依据,建好索引或算好 level,再靠递归自然展开。不复杂但容易忽略 context 切换和 key 的声明位置。










