在 SQL 中,使用 聚类(GROUP BY)
后将某个字段的数据串联为一个字符串,常见的方法包括以下几种,取决于数据库管理系统(DBMS)的具体支持功能:
1. 使用 GROUP_CONCAT
(MySQL 等支持)
GROUP_CONCAT
是 MySQL 提供的函数,用于将分组中的字段值串联成字符串。
SELECT group_column, GROUP_CONCAT(field_to_concatenate SEPARATOR ',') AS concatenated_string
FROM table_name
GROUP BY group_column;
SEPARATOR
参数 :定义连接的分隔符(默认为逗号,
)。- 支持系统:MySQL、MariaDB。
2. 使用 STRING_AGG
(SQL Server、PostgreSQL 等支持)
STRING_AGG
是更现代化的函数,用于将字段值连接为一个字符串。
-- PostgreSQL / SQL Server
SELECT group_column, STRING_AGG(field_to_concatenate, ',') AS concatenated_string
FROM table_name
GROUP BY group_column;
STRING_AGG
语法 :- 第一个参数是要连接的字段。
- 第二个参数是分隔符。
- 支持系统:PostgreSQL、SQL Server(2017+)。
3. 使用 XML 或 JSON 方法(SQL Server)
在 SQL Server 中,也可以使用 XML 路径或者 JSON 的方法进行字符串连接。
XML PATH 方法
SELECT group_column, STUFF( (SELECT ',' + field_to_concatenate
FROM table_name t2
WHERE t2.group_column = t1.group_column
FOR XML PATH('')), 1, 1, '') AS concatenated_string
FROM table_name t1
GROUP BY group_column;
- 原理 :利用
FOR XML PATH('')
将数据生成无标签的 XML,然后用STUFF
去掉前导逗号。
JSON PATH 方法(SQL Server 2017+)
SELECT group_column, STRING_AGG(field_to_concatenate, ',') AS concatenated_string
FROM table_name
GROUP BY group_column;
4. 递归 CTE 或用户定义函数(适合不支持内置串联函数的数据库)
对于不支持 GROUP_CONCAT
或 STRING_AGG
的数据库,可以通过递归 CTE 或用户定义的函数实现。
示例:递归 CTE(SQL Server)
WITH CTE AS
(
SELECT group_column,
field_to_concatenate AS concatenated_string,
ROW_NUMBER() OVER (PARTITION BY group_column ORDER BY id) AS row_num
FROM table_name
UNION ALL
SELECT cte.group_column,
cte.concatenated_string + ',' + t.field_to_concatenate,
cte.row_num + 1
FROM CTE cte
JOIN table_name t
ON cte.group_column = t.group_column AND
t.row_num = cte.row_num + 1
)
SELECT group_column,
MAX(concatenated_string) AS concatenated_string
FROM CTE
GROUP BY group_column;
5. 手动拼接(通过程序语言处理)
如果数据库本身不支持上述方法,可以在程序端(如 Python、JavaScript、PHP 等)处理分组并拼接字符串。
总结
- 推荐方法 :尽量使用 DBMS 内置的函数(如
GROUP_CONCAT
或STRING_AGG
),实现简单高效。 - 兼容性 :
- MySQL、MariaDB:
GROUP_CONCAT
- PostgreSQL、SQL Server(2017+):
STRING_AGG
- SQL Server(旧版本):
XML PATH
- 其他数据库:可以考虑递归 CTE 或程序端处理。
- MySQL、MariaDB: