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

XSLT 2.0 的 group-by 是 xsl:for-each-group 的核心分组机制,不是独立标签,必须配合使用。 它按表达式计算出的值对节点序列进行逻辑分组,相同值的相邻或非相邻节点会被归入同一组(取决于你选哪个分组属性)。关键在于:它不改变原始顺序,只组织遍历逻辑。
group-by 属性的基本用法
它用于指定“按什么值分组”,表达式结果作为分组键。每个匹配节点都会计算该表达式,值相等的节点归为一组。
- 必须写在
开始标签内,且select属性是必需的(指定要处理的节点集) - 表达式可以是简单路径(如
@category)、函数调用(如upper-case(@type))或任意合法 XPath 2.0 表达式 - 分组键支持任意类型(字符串、数字、日期、QName 等),自动按类型语义比较
- 空序列(
())或未定义值会被统一视为一个特殊分组键
和 group-adjacent 的区别很实际
初学者常混淆 group-by 和 group-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 的简洁性远超手写递归或多次遍历——关键是理解它“一次扫描、动态建组”的设计逻辑。










