- GROUP_CONCAT(xxx): Es para concatenar las cadenas correspondientes entre paréntesis en el grupo. Si el parámetro xxx entre paréntesis en el grupo tiene varias líneas, entonces las cadenas de estas líneas se concatenarán, y habrá una cadena entre cada cadena separada por símbolos específicos.
Formato gramatical correspondiente
# 将分组中column1这一列对应的多行的值按照column2 升序或者降序进行连接,其中分隔符为seq
# 如果用到了DISTINCT,将表示将不重复的column1按照column2升序或者降序连接
# 如果没有指定SEPARATOR的话,也就是说没有写,那么就会默认以 ','分隔
GROUP_CONCAT([DISTINCT] column1 [ORDER BY column2 ASC\DESC] [SEPARATOR seq]);
[ORDER BY column2 ASC\DESC]: indica que se conectará en orden ascendente o descendente de acuerdo con la columna 2. La columna 2 no requiere necesariamente la columna 1, siempre que la columna 2 esté en este grupo . Si no se escribe una oración ORDER BY, entonces el La conexión no es secuencial .
[ SEPARATOR seq] : Indica con qué separador se separará cada columna 1. Por ejemplo,
, SEPARATOR ' ' significa que se separará la columna 1. Si no se especifica seq, es decir, no se escribe el segmento SEPARATOR seq, entonces se separará por defecto .
Los datos que se conectarán en la función CONCAT contienen NULL, y el retorno final es NULL, pero GROUP_CONCAT no hará esto, ignorará el valor NULL.
mysql> SELECT * FROM employee2;
+----+-----------+------+---------+---------+
| id | name | age | salary | dept_id |
+----+-----------+------+---------+---------+
| 3 | 小肖 | 29 | 30000.0 | 1 |
| 4 | 小东 | 30 | 40000.0 | 2 |
| 6 | 小非 | 24 | 23456.0 | 3 |
| 7 | 晓飞 | 30 | 15000.0 | 4 |
| 8 | 小林 | 23 | 24000.0 | NULL |
| 10 | 小五 | 20 | 4500.0 | NULL |
| 11 | 张山 | 24 | 40000.0 | 1 |
| 12 | 小肖 | 28 | 35000.0 | 2 |
| 13 | 李四 | 23 | 50000.0 | 1 |
| 17 | 王武 | 24 | 56000.0 | 2 |
| 18 | 猪小屁 | 2 | 56000.0 | 2 |
| 19 | 小玉 | 25 | 58000.0 | 1 |
| 21 | 小张 | 23 | 50000.0 | 1 |
| 22 | 小胡 | 25 | 25000.0 | 2 |
| 96 | 小肖 | 19 | 35000.0 | 1 |
| 97 | 小林 | 20 | 20000.0 | 2 |
+----+-----------+------+---------+---------+
16 rows in set (0.16 sec)
mysql> SELECT
-> dept_id,
-> GROUP_CONCAT(name ORDER BY age DESC SEPARATOR '*') -- 分组中的name中的多行数据将按照age降序进行连接,分隔符为 *
-> FROM employee2
-> GROUP BY dept_id; -- 注意如果这里没有GROUP BY dept_id,那么就会因为输出dept_id而发生报错
+---------+----------------------------------------------------+
| dept_id | GROUP_CONCAT(name ORDER BY age DESC SEPARATOR '*') |
+---------+----------------------------------------------------+
| NULL | 小林*小五 |
| 1 | 小肖*小玉*张山*小张*李四*小肖 |
| 2 | 小东*小肖*小胡*王武*小林*猪小屁 |
| 3 | 小非 |
| 4 | 晓飞 |
+---------+----------------------------------------------------+
mysql> SELECT
-> GROUP_CONCAT(name SEPARATOR '*')
-> FROM employee2; -- 这时候虽然没有使用GROUP BY,但是可以正常运行,此时是将所有的name连接,连接时为无序,分隔符为*
+-------------------------------------------------------------------------------------------------------+
| GROUP_CONCAT(name SEPARATOR '*') |
+--------------------------------------------------------------------------------------------------------
| 小肖*小东*小非*晓飞*小林*小五*张山*小肖*李四*王武*猪小屁*小玉*小张*小胡*小肖*小林 |
+-------------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)
mysql> SELECT
-> GROUP_CONCAT(DISTINCT name SEPARATOR '*') -- 将不同的name进行连接
-> FROM employee2;
+-----------------------------------------------------------------------------------------------+
| GROUP_CONCAT(DISTINCT name SEPARATOR '*') |
+-----------------------------------------------------------------------------------------------+
| 小东*小五*小张*小林*小玉*小肖*小胡*小非*张山*晓飞*李四*猪小屁*王武 |
+-----------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)
mysql> SELECT
-> dept_id,
-> GROUP_CONCAT(name) AS employees
-> FROM employee2
-> GROUP BY dept_id; -- 输出每个部门的员工,每个员工之间用逗号分隔,因为没有写SEPARATOR ,所以就默认以逗号分隔
+---------+----------------------------------------------+
| dept_id | employees |
+---------+----------------------------------------------+
| NULL | 小林,小五 |
| 1 | 小肖,张山,李四,小玉,小张,小肖 |
| 2 | 小东,小肖,王武,猪小屁,小胡,小林 |
| 3 | 小非 |
| 4 | 晓飞 |
+---------+----------------------------------------------+
5 rows in set (0.00 sec)
mysql> SELECT
-> dept_id,
-> GROUP_CONCAT(DISTINCT name) -- 将不同的name连接,并且用逗号分隔
-> AS employees
-> FROM employee2
-> GROUP BY dept_id;
+---------+----------------------------------------------+
| dept_id | employees |
+---------+----------------------------------------------+
| NULL | 小五,小林 |
| 1 | 小张,小玉,小肖,张山,李四 |
| 2 | 小东,小林,小肖,小胡,猪小屁,王武 |
| 3 | 小非 |
| 4 | 晓飞 |
+---------+----------------------------------------------+
5 rows in set (0.00 sec)
Entonces, al usar GROUP_CONCAT, puede resolver el problema de resumir de acuerdo con dept_no .
Vale la pena mencionar que GROUP_CONCAT solo conecta múltiples filas de datos en la columna xxx en una fila de cadenas, mientras que CONCAT puede conectar múltiples columnas de datos. Entonces, cuando GROUP_CONCAT se usa junto con (CONCAT_WS o CONCAT), hace que la consulta sea más eficiente . Por ejemplo, el siguiente ejemplo es exactamente el caso:
mysql> SELECT * FROM employee2;
+----+-----------+------+---------+---------+
| id | name | age | salary | dept_id |
+----+-----------+------+---------+---------+
| 3 | 小肖 | 29 | 30000.0 | 1 |
| 4 | 小东 | 30 | 40000.0 | 2 |
| 6 | 小非 | 24 | 23456.0 | 3 |
| 7 | 晓飞 | 30 | 15000.0 | 4 |
| 8 | 小林 | 23 | 24000.0 | NULL |
| 10 | 小五 | 20 | 4500.0 | NULL |
| 11 | 张山 | 24 | 40000.0 | 1 |
| 12 | 小肖 | 28 | 35000.0 | 2 |
| 13 | 李四 | 23 | 50000.0 | 1 |
| 17 | 王武 | 24 | 56000.0 | 2 |
| 18 | 猪小屁 | 2 | 56000.0 | 2 |
| 19 | 小玉 | 25 | 58000.0 | 1 |
| 21 | 小张 | 23 | 50000.0 | 1 |
| 22 | 小胡 | 25 | 25000.0 | 2 |
| 96 | 小肖 | 19 | 35000.0 | 1 |
| 97 | 小林 | 20 | 20000.0 | 2 |
+----+-----------+------+---------+---------+
16 rows in set (0.00 sec)
mysql> SELECT
-> dept_id,
-> GROUP_CONCAT(CONCAT(name,"(",salary,")") SEPARATOR ';') -- 将利用name(salary)进行连接,其中分隔符为';'
-> FROM
-> employee2
-> GROUP BY dept_id;
+---------+---------------------------------------------------------------------------------------------+
| dept_id | GROUP_CONCAT(CONCAT(name,"(",salary,")") SEPARATOR ';')
+---------+---------------------------------------------------------------------------------------------+
| NULL | 小林(24000.0);小五(4500.0)
| 1 | 小肖(30000.0);张山(40000.0);李四(50000.0);小玉(58000.0);小张(50000.0);小肖(35000.0)
| 2 | 小东(40000.0);小肖(35000.0);王武(56000.0);猪小屁(56000.0);小胡(25000.0);小林(20000.0)
| 3 | 小非(23456.0)
| 4 | 晓飞(15000.0)
+---------+---------------------------------------------------------------------------------------------+
5 rows in set (0.30 sec)
Vea este artículo para más detalles: MySQL group_concat()