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

XSLT 可以通过识别层级标识(比如 level、parent-id、缩进空格、序号前缀等)把扁平 XML 数据构建成嵌套的父子结构。核心思路是:用 递归模板 + 条件匹配 + 上下文定位,而不是靠循环。
如果源数据每个节点带 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 为空或为 0),适合用 键(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() 去掉点号,统计层级深度注意:XSLT 1.0 不支持正则,复杂模式建议先用外部工具预处理;XSLT 2.0+ 可用 analyze-string 或 tokenize() 更稳妥。
xsl:for-each 嵌套多层模拟树——性能差且难维护xsl:call-template)current() 而非 . 在谓词里,防止上下文错位基本上就这些。关键是选对层级依据,建好索引或算好 level,再靠递归自然展开。不复杂但容易忽略 context 切换和 key 的声明位置。
以上就是XSLT怎么把扁平数据转换成父子结构的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号