XSLT 2.0 的 group-by 是 xsl:for-each-group 的分组属性,按表达式值全局分组节点,不改变原始顺序;需配合 select 使用,支持任意类型键值,区别于仅分邻近节点的 group-adjacent。
XSLT 2.0 的 group-by 是 xsl:for-each-group 的核心分组机制,不是独立标签,必须配合使用。 它按表达式计算出的值对节点序列进行逻辑分组,相同值的相邻或非相邻节点会被归入同一组(取决于你选哪个分组属性)。关键在于:它不改变原始顺序,只组织遍历逻辑。
它用于指定“按什么值分组”,表达式结果作为分组键。每个匹配节点都会计算该表达式,值相等的节点归为一组。
开始标签内,且 select 属性是必需的(指定要处理的节点集)@category)、函数调用(如 upper-case(@type))或任意合法 XPath 2.0 表达式
列(())或未定义值会被统一视为一个特殊分组键初学者常混淆 group-by 和 group-adjacent —— 本质区别就一条:group-by 全局匹配,group-adjacent 只认紧挨着的相同值。
,再一个 ,再两个
group-by="@type" → 得到 2 组:A(5个节点)、B(1个节点)group-adjacent="@type" → 得到 3 组:A(3个)、B(1个)、A(2个)group-by;做段落合并、连续样式控制用 group-adjacent
进入 内部后,这两个函数就是你的分组工具箱:
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 的简洁性远超手写递归或多次遍历——关键是理解它“一次扫描、动态建组”的设计逻辑。