Результаты MySQL в виде списка, разделенного запятыми

129

Мне нужно выполнить такой запрос:

SELECT p.id, p.name, 
       (SELECT name 
          FROM sites s 
         WHERE s.id = p.site_id) AS site_list
  FROM publications p

Но я бы хотел, чтобы подвыбор возвращал список, разделенный запятыми, а не столбец данных. Возможно ли это вообще, и если да, то как?

Глен Солсберри
источник

Ответы:

250

Вы можете использовать GROUP_CONCAT для этого, например, что-то вроде

SELECT p.id, p.name, GROUP_CONCAT(s.name) AS site_list
FROM sites s
INNER JOIN publications p ON(s.id = p.site_id)
GROUP BY p.id, p.name;
Пол Диксон
источник
10
Также обратите внимание: если вы используете PHPMyAdmin и хотите вывести на страницу список, разделенный запятыми, используйте, GROUP_CONCAT(CAST(s.name AS CHAR))иначе он просто вернет что-то совершенно бесполезное, например [BLOB - 20 Bytes].
devios1 01
3
Намерение в порядке, и MySQL позволяет это, но будьте осторожны (как правило) с использованием GROUP BY. Элементы в списке выбора должны быть действительными агрегатами в контексте предложения GROUP BY. В этом случае p.name не является строго допустимым. Любая база данных, соответствующая стандарту SQL, будет рассматривать это как ошибку. В этом случае либо используйте MAX (p.name) в списке выбора, либо добавьте p.name в предложение GROUP BY. Поскольку Пол, вероятно, имеет в виду p.id для представления первичного или уникального ключа, добавление p.name в предложение GROUP BY не повлияет на конечный результат.
Джон Армстронг - Xgc
Обратите внимание, что вам может потребоваться установить максимальную длину сеанса для каждого stackoverflow.com/questions/2567000/…
sobelito
Огромное спасибо! Вы мне очень помогли!
Андре Агостиньо
11

Вместо использования group concat()вы можете использовать простоconcat()

Select concat(Col1, ',', Col2) as Foo_Bar from Table1;

изменить это работает только в mySQL; Oracle concat принимает только два аргумента. В oracle вы можете использовать что-то вроде select col1 || ',' || col2 || ',' || col3 как foobar из table1; на сервере sql вы использовали бы + вместо каналов.

Роберт Куинн
источник
2
Это не должно работать в случае GROUP BY, в то время как GROUP_CONCAT () объединит содержимое одного столбца
Арам Пароникян
6

Только сейчас я столкнулся с этой ситуацией и нашел еще несколько интересных особенностей вокруг GROUP_CONCAT . Надеюсь, эти детали заставят вас почувствовать себя интересными.

простой GROUP_CONCAT

SELECT GROUP_CONCAT(TaskName) 
FROM Tasks;

Результат:

+------------------------------------------------------------------+
| GROUP_CONCAT(TaskName)                                           |
+------------------------------------------------------------------+
| Do garden,Feed cats,Paint roof,Take dog for walk,Relax,Feed cats |
+------------------------------------------------------------------+

GROUP_CONCAT с DISTINCT

SELECT GROUP_CONCAT(TaskName) 
FROM Tasks;

Результат:

+------------------------------------------------------------------+
| GROUP_CONCAT(TaskName)                                           |
+------------------------------------------------------------------+
| Do garden,Feed cats,Paint roof,Take dog for walk,Relax,Feed cats |
+------------------------------------------------------------------+

GROUP_CONCAT с DISTINCT и ORDER BY

SELECT GROUP_CONCAT(DISTINCT TaskName ORDER BY TaskName DESC) 
FROM Tasks;

Результат:

+--------------------------------------------------------+
| GROUP_CONCAT(DISTINCT TaskName ORDER BY TaskName DESC) |
+--------------------------------------------------------+
| Take dog for walk,Relax,Paint roof,Feed cats,Do garden |
+--------------------------------------------------------+

GROUP_CONCAT с DISTINCT и SEPARATOR

SELECT GROUP_CONCAT(DISTINCT TaskName SEPARATOR ' + ') 
FROM Tasks;

Результат:

+----------------------------------------------------------------+
| GROUP_CONCAT(DISTINCT TaskName SEPARATOR ' + ')                |
+----------------------------------------------------------------+
| Do garden + Feed cats + Paint roof + Relax + Take dog for walk |
+----------------------------------------------------------------+

GROUP_CONCAT и объединение столбцов

SELECT GROUP_CONCAT(TaskId, ') ', TaskName SEPARATOR ' ') 
FROM Tasks;

Результат:

+------------------------------------------------------------------------------------+
| GROUP_CONCAT(TaskId, ') ', TaskName SEPARATOR ' ')                                 |
+------------------------------------------------------------------------------------+
| 1) Do garden 2) Feed cats 3) Paint roof 4) Take dog for walk 5) Relax 6) Feed cats |
+------------------------------------------------------------------------------------+

GROUP_CONCAT и сгруппированные результаты Предположим, что следующие результаты перед использованиемGROUP_CONCAT

+------------------------+--------------------------+
| ArtistName             | AlbumName                |
+------------------------+--------------------------+
| Iron Maiden            | Powerslave               |
| AC/DC                  | Powerage                 |
| Jim Reeves             | Singing Down the Lane    |
| Devin Townsend         | Ziltoid the Omniscient   |
| Devin Townsend         | Casualties of Cool       |
| Devin Townsend         | Epicloud                 |
| Iron Maiden            | Somewhere in Time        |
| Iron Maiden            | Piece of Mind            |
| Iron Maiden            | Killers                  |
| Iron Maiden            | No Prayer for the Dying  |
| The Script             | No Sound Without Silence |
| Buddy Rich             | Big Swing Face           |
| Michael Learns to Rock | Blue Night               |
| Michael Learns to Rock | Eternity                 |
| Michael Learns to Rock | Scandinavia              |
| Tom Jones              | Long Lost Suitcase       |
| Tom Jones              | Praise and Blame         |
| Tom Jones              | Along Came Jones         |
| Allan Holdsworth       | All Night Wrong          |
| Allan Holdsworth       | The Sixteen Men of Tain  |
+------------------------+--------------------------+
USE Music;
SELECT ar.ArtistName,
    GROUP_CONCAT(al.AlbumName)
FROM Artists ar
INNER JOIN Albums al
ON ar.ArtistId = al.ArtistId
GROUP BY ArtistName;

Результат:

+------------------------+----------------------------------------------------------------------------+
| ArtistName             | GROUP_CONCAT(al.AlbumName)                                                 |
+------------------------+----------------------------------------------------------------------------+
| AC/DC                  | Powerage                                                                   |
| Allan Holdsworth       | All Night Wrong,The Sixteen Men of Tain                                    |
| Buddy Rich             | Big Swing Face                                                             |
| Devin Townsend         | Epicloud,Ziltoid the Omniscient,Casualties of Cool                         |
| Iron Maiden            | Somewhere in Time,Piece of Mind,Powerslave,Killers,No Prayer for the Dying |
| Jim Reeves             | Singing Down the Lane                                                      |
| Michael Learns to Rock | Eternity,Scandinavia,Blue Night                                            |
| The Script             | No Sound Without Silence                                                   |
| Tom Jones              | Long Lost Suitcase,Praise and Blame,Along Came Jones                       |
+------------------------+----------------------------------------------------------------------------+
Ганеса Виджаякумар
источник
3

В моем случае я должен объединить весь номер счета человека, номер мобильного телефона которого уникален. Поэтому для этого я использовал следующий запрос.

SELECT GROUP_CONCAT(AccountsNo) as Accounts FROM `tblaccounts` GROUP BY MobileNumber

Результат запроса ниже:

Accounts
93348001,97530801,93348001,97530801
89663501
62630701
6227895144840002
60070021
60070020
60070019
60070018
60070017
60070016
60070015
Винеш Чиннайян
источник