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>
调用时传入全部节点和起始 level 即可:<call-template name="build-tree"><with-param name="nodes" select="/data/item"></with-param></call-template>
用 parent-id 关联父子关系
若数据含 parent-id(根节点 parent-id 为空或为 0),适合用 键(key)预索引 提升效率:
- 先定义 key:
<key name="children" match="item" use="parent-id"></key> - 对根节点(
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 的声明位置。






























