XSLT 2.0中的group-by怎么用 分组功能详解

14次阅读

XSLT 2.0 的 group-by 是 xsl:for-each-group 的分组属性,按表达式值全局分组节点,不改变原始顺序;需配合 select 使用,支持任意类型键值,区别于仅分邻近节点的 group-adjacent。

XSLT 2.0中的group-by怎么用 分组功能详解

XSLT 2.0 的 group-byxsl:for-each-group 的核心分组机制,不是独立标签,必须配合使用。 它按表达式计算出的值对节点序列进行逻辑分组,相同值的相邻或非相邻节点会被归入同一组(取决于你选哪个分组属性)。关键在于:它不改变原始顺序,只组织遍历逻辑。

group-by 属性的基本用法

它用于指定“按什么值分组”,表达式结果作为分组键。每个匹配节点都会计算该表达式,值相等的节点归为一组。

  • 必须写在 开始标签内,且 select 属性是必需的(指定要处理的节点集)
  • 表达式可以是简单路径(如 @category)、函数调用(如 upper-case(@type))或任意合法 XPath 2.0 表达式
  • 分组键支持任意类型(字符串、数字、日期、QName 等),自动按类型语义比较
  • 空序列(())或未定义值会被统一视为一个特殊分组键

和 group-adjacent 的区别很实际

初学者常混淆 group-bygroup-adjacent —— 本质区别就一条:group-by 全局匹配,group-adjacent 只认紧挨着的相同值。

  • 比如 xml 中连续三个 ,再一个 ,再两个
  • group-by="@type" → 得到 2 组:A(5个节点)、B(1个节点)
  • group-adjacent="@type" → 得到 3 组:A(3个)、B(1个)、A(2个)
  • 所以做目录、分类汇总用 group-by;做段落合并、连续样式控制用 group-adjacent

配合 current-group() 和 current-grouping-key() 使用

进入 内部后,这两个函数就是你的分组工具箱:

  • current-group() 返回当前这组的所有节点(保持原始顺序),可对其用 count()sum()for-each 等操作
  • current-grouping-key() 返回本次分组所依据的值(即 group-by 表达式的结果),常用于输出标题或条件判断
  • 例如:(共 条)

常见组合与注意事项

group-by 不是万能钥匙,得看场景选对搭档:

  • 想按层级结构分组(如先按年、再按月)?不能嵌套 for-each-group,改用 group-starting-with 或多级 group-by 表达式(如 concat(@year, '-', @month)
  • 需要忽略大小写或空白?在 group-by 表达式里预处理:normalize-space(upper-case(@name))
  • 分组后要排序?加 子元素,它作用于每组内部,不影响分组逻辑
  • 注意性能:复杂表达式会在每个节点上重复计算,高频场景建议提前用 xsl:variable 缓存中间结果

基本上就这些。用熟了你会发现,group-by 的简洁性远超手写递归或多次遍历——关键是理解它“一次扫描、动态建组”的设计逻辑。

text=ZqhQzanResources