MySQL: GROUP_CONCAT с левым соединением - PullRequest
50 голосов
/ 16 декабря 2010

У меня проблема с функцией MySQL «GROUP_CONCAT».Я проиллюстрирую мою проблему, используя простую базу данных службы поддержки:

CREATE TABLE Tickets (
 id INTEGER NOT NULL PRIMARY KEY,
 requester_name VARCHAR(255) NOT NULL,
 description TEXT NOT NULL);

CREATE TABLE Solutions (
 id INTEGER NOT NULL PRIMARY KEY,
 ticket_id INTEGER NOT NULL,
 technician_name VARCHAR(255) NOT NULL,
 solution TEXT NOT NULL,
 FOREIGN KEY (ticket_id) REFERENCES Tickets.id);

INSERT INTO Tickets VALUES(1, 'John Doe', 'My computer is not booting.');
INSERT INTO Tickets VALUES(2, 'Jane Doe', 'My browser keeps crashing.');
INSERT INTO Solutions VALUES(1, 1, 'Technician A', 'I tried to solve this but was unable to. I will pass this on to Technician B since he is more experienced than I am.');
INSERT INTO Solutions VALUES(2, 1, 'Technician B', 'I reseated the RAM and that fixed the problem.');
INSERT INTO Solutions VALUES(3, 2, 'Technician A', 'I was unable to figure this out. I will again pass this on to Technician B.');
INSERT INTO Solutions VALUES(4, 2, 'Technician B', 'I re-installed the browser and that fixed the problem.');

Обратите внимание, что в этой базе данных службы поддержки есть две заявки, каждая с двумя записями решения.Моя цель - использовать оператор SELECT для создания списка всех заявок в базе данных с соответствующими записями решения.Это оператор SELECT, который я использую:

SELECT Tickets.*, GROUP_CONCAT(Solutions.solution) AS CombinedSolutions
FROM Tickets
LEFT JOIN Solutions ON Tickets.id = Solutions.ticket_id
ORDER BY Tickets.id;

Проблема с приведенным выше оператором SELECT заключается в том, что он возвращает только одну строку:

id: 1
requester_name: John Doe
description: My computer is not booting.
CombinedSolutions: I tried to solve this but was unable to. I will pass this on to Technician B since he is more experienced than I am.,I reseated the RAM and that fixed the problem.,I was unable to figure this out. I will again pass this on to Technician B.,I re-installed the browser and that fixed the problem.

Обратите внимание, что он возвращает информацию билета 1 си записи решения для билета 1 и билета 2.

Что я делаю не так?Спасибо!

Ответы [ 3 ]

85 голосов
/ 16 декабря 2010

Использование:

   SELECT t.*,
          x.combinedsolutions
     FROM TICKETS t
LEFT JOIN (SELECT s.ticket_id,
                  GROUP_CONCAT(s.soution) AS combinedsolutions
             FROM SOLUTIONS s 
         GROUP BY s.ticket_id) x ON x.ticket_id = t.ticket_id

Альтернатива:

   SELECT t.*,
          (SELECT GROUP_CONCAT(s.soution)
             FROM SOLUTIONS s 
            WHERE s.ticket_id = t.ticket_id) AS combinedsolutions
     FROM TICKETS t
2 голосов
/ 03 июня 2016

Я думаю, что комментарий @Dylan Valade является самым простым ответом, поэтому я публикую его как другой ответ: простое добавление GROUP BY Tickets.id в SELECT OP должно решить проблему.Это исправило мою собственную проблему.

Однако, для баз данных, которые не являются небольшими, принятый ответ, особенно если есть какие-то предикаты на Tickets.id, по-видимому, не включают в себя полное сканирование таблицы и поэтому, пока предыдущий абзац возвращаетправильные результаты в моем случае оказываются гораздо менее эффективными.

1 голос
/ 14 апреля 2019

Вам просто нужно добавить GROUP_BY:

SELECT Tickets.*, GROUP_CONCAT(Solutions.solution) AS CombinedSolutions FROM Tickets 
LEFT JOIN Solutions ON Tickets.id = Solutions.ticket_id 
GROUP_BY Tickets.id 
ORDER BY Tickets.id;
...