MySQL GROUP BY две колонки - PullRequest
       7

MySQL GROUP BY две колонки

57 голосов
/ 02 февраля 2010

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

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

SELECT clients.id, clients.name, portfolios.id, SUM ( portfolios.portfolio +  portfolios.cash ) AS total
FROM clients, portfolios
WHERE clients.id = portfolios.client_id
GROUP BY portfolios.id, clients.id
ORDER BY total DESC
LIMIT 30 

Ответы [ 2 ]

143 голосов
/ 02 февраля 2010

Во-первых, давайте сделаем несколько тестовых данных:

create table client (client_id integer not null primary key auto_increment,
                     name varchar(64));
create table portfolio (portfolio_id integer not null primary key auto_increment,
                        client_id integer references client.id,
                        cash decimal(10,2),
                        stocks decimal(10,2));
insert into client (name) values ('John Doe'), ('Jane Doe');
insert into portfolio (client_id, cash, stocks) values (1, 11.11, 22.22),
                                                       (1, 10.11, 23.22),
                                                       (2, 30.30, 40.40),
                                                       (2, 40.40, 50.50);

Если вам не нужен идентификатор портфеля, это будет легко:

select client_id, name, max(cash + stocks)
from client join portfolio using (client_id)
group by client_id

+-----------+----------+--------------------+
| client_id | name     | max(cash + stocks) |
+-----------+----------+--------------------+
|         1 | John Doe |              33.33 | 
|         2 | Jane Doe |              90.90 | 
+-----------+----------+--------------------+

Поскольку вам нужен идентификатор портфеля, все становится сложнее. Давайте сделаем это по шагам. Сначала мы напишем подзапрос, который возвращает максимальное значение портфеля для каждого клиента:

select client_id, max(cash + stocks) as maxtotal
from portfolio
group by client_id

+-----------+----------+
| client_id | maxtotal |
+-----------+----------+
|         1 |    33.33 | 
|         2 |    90.90 | 
+-----------+----------+

Затем мы запросим таблицу портфелей, но используем соединение с предыдущим подзапросом, чтобы сохранить только те портфели, суммарное значение которых является максимальным для клиента:

 select portfolio_id, cash + stocks from portfolio 
 join (select client_id, max(cash + stocks) as maxtotal 
       from portfolio
       group by client_id) as maxima
 using (client_id)
 where cash + stocks = maxtotal

+--------------+---------------+
| portfolio_id | cash + stocks |
+--------------+---------------+
|            5 |         33.33 | 
|            6 |         33.33 | 
|            8 |         90.90 | 
+--------------+---------------+

Наконец, мы можем присоединиться к клиентской таблице (как вы сделали), чтобы включить имя каждого клиента:

select client_id, name, portfolio_id, cash + stocks
from client
join portfolio using (client_id)
join (select client_id, max(cash + stocks) as maxtotal
      from portfolio 
      group by client_id) as maxima
using (client_id)
where cash + stocks = maxtotal

+-----------+----------+--------------+---------------+
| client_id | name     | portfolio_id | cash + stocks |
+-----------+----------+--------------+---------------+
|         1 | John Doe |            5 |         33.33 | 
|         1 | John Doe |            6 |         33.33 | 
|         2 | Jane Doe |            8 |         90.90 | 
+-----------+----------+--------------+---------------+

Обратите внимание, что это возвращает две строки для Джона Доу, потому что у него есть два портфеля с одинаковой общей стоимостью. Чтобы избежать этого и выбрать произвольный топ-портфель, пометьте предложение GROUP BY:

select client_id, name, portfolio_id, cash + stocks
from client
join portfolio using (client_id)
join (select client_id, max(cash + stocks) as maxtotal
      from portfolio 
      group by client_id) as maxima
using (client_id)
where cash + stocks = maxtotal
group by client_id, cash + stocks

+-----------+----------+--------------+---------------+
| client_id | name     | portfolio_id | cash + stocks |
+-----------+----------+--------------+---------------+
|         1 | John Doe |            5 |         33.33 | 
|         2 | Jane Doe |            8 |         90.90 | 
+-----------+----------+--------------+---------------+
92 голосов
/ 22 мая 2015

Использование Concat для группы по будет работать

SELECT clients.id, clients.name, portfolios.id, SUM ( portfolios.portfolio + portfolios.cash ) AS total
FROM clients, portfolios
WHERE clients.id = portfolios.client_id
GROUP BY CONCAT(portfolios.id, "-", clients.id)
ORDER BY total DESC
LIMIT 30
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...