如何用 STRING_AGG / GROUP_CONCAT 实现列转行(聚合字符串)

14次阅读

用 STRING_AGG 或 GROUP_CONCAT 可拼接多行字符串,但默认行为、分隔符、NULL 处理和排序差异大;GROUP_CONCAT 无匹配行时返回 NULL,字段为 NULL 时被忽略,需用 COALESCE 防空;mysql 默认长度限 1024 字节;STRING_AGG 的 ORDER BY 必须在括号内;postgresql 支持 DISTINCT,SQL Server 不支持;跨库语法不兼容,ORM 中优先使用其聚合接口

如何用 STRING_AGG / GROUP_CONCAT 实现列转行(聚合字符串)

直接说结论:STRING_AGG(PostgreSQL / SQL Server 2017+)或 GROUP_CONCAT(MySQL)能干净地把多行值拼成一行字符串,但默认行为、分隔符、NULL 处理和排序控制差异很大,不注意就会漏数据或顺序错乱。

为什么 GROUP_CONCAT 有时不返回任何结果?

常见于 WHERE 条件过滤后没匹配到行 —— GROUP_CONCAT聚合函数,若分组内无记录,结果为 NULL(不是空字符串)。另外,若字段本身为 NULL,默认被忽略,整段拼接可能“消失”。

  • IFNULL(..., '')COALESCE(..., '') 包裹整个表达式防空
  • GROUP_CONCAT(col SEPARATOR ',' ORDER BY col) 显式指定排序,否则顺序不确定
  • MySQL 默认最大长度是 1024 字节group_concat_max_len),超长会被截断,查不到后半部分

STRING_AGG 的 ORDER BY 必须写在括号里,不能放外面

很多人写成 STRING_AGG(name) ORDER BY name,这是错的 —— ORDER BYSTRING_AGG 的参数,必须紧贴函数内部,正确写法是:STRING_AGG(name, ', ' ORDER BY name)。SQL Server 和 PostgreSQL 都遵循这个语法。

  • 分隔符是第二个参数,不可省略;想用空字符串就写 ''
  • PostgreSQL 支持 STRING_AGG(DISTINCT name, ', ') 去重,SQL Server 不支持 DISTINCT 关键字
  • 遇到 NULL 值时,STRING_AGG 默认跳过,和 GROUP_CONCAT 一致,无需额外处理

数据库写法兼容性差,别硬套一个 SQL 跑所有库

MySQL 没有 STRING_AGG,PostgreSQL/SQL Server 也不认 GROUP_CONCAT。连分隔符参数位置都不同:MySQL 是第二参数,PostgreSQL 是第二参数但带 ORDER BY 子句,SQL Server 同样。

  • MySQL:GROUP_CONCAT(name ORDER BY id SEPARATOR ';')
  • PostgreSQL:STRING_AGG(name, '; ' ORDER BY id)
  • SQL Server:STRING_AGG(name, '; ') WITHIN GROUP (ORDER BY id)
  • 如果用 ORM(如 SQLAlchemy、Django ORM),优先用其聚合接口,别手拼原生 SQL

最常被忽略的是排序依赖 —— 没写 ORDER BY 时,拼出来的字符串顺序完全由查询执行计划决定,可能每次都不一样。哪怕只是临时查数据,也建议显式加上。

text=ZqhQzanResources