XSLT怎么把扁平数据转换成父子结构

8次阅读

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

XSLT 怎么把扁平数据转换成父子结构

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-stringtokenize() 更稳妥。

避免常见坑

  • 别用 xsl:for-each 嵌套多层模拟树——性能差且难维护
  • 递归必须有明确终止条件(如无子节点时跳过 xsl:call-template
  • current() 而非 . 在谓词里,防止上下文错位
  • 测试时用小样本,关注生成的嵌套深度是否符合预期

基本上就这些。关键是选对层级依据,建好索引或算好 level,再靠递归自然展开。不复杂但容易忽略 context 切换和 key 的声明位置。

text=ZqhQzanResources