MySQL запрос, MAX () + GROUP BY - PullRequest
       18

MySQL запрос, MAX () + GROUP BY

26 голосов
/ 14 апреля 2011

Дафт SQL вопрос. У меня есть таблица, как так ('PID' является основным шагом автоинкремент)

CREATE TABLE theTable (
    `pid` INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
    `timestamp` TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
    `cost` INT UNSIGNED NOT NULL,
    `rid` INT NOT NULL,
) Engine=InnoDB;

Фактические данные таблицы:

INSERT INTO theTable (`pid`, `timestamp`, `cost`, `rid`)
VALUES
  (1, '2011-04-14 01:05:07', 1122, 1),
  (2, '2011-04-14 00:05:07', 2233, 1),
  (3, '2011-04-14 01:05:41', 4455, 2),
  (4, '2011-04-14 01:01:11', 5566, 2),
  (5, '2011-04-14 01:06:06', 345, 1),
  (6, '2011-04-13 22:06:06', 543, 2),
  (7, '2011-04-14 01:14:14', 5435, 3),
  (8, '2011-04-14 01:10:13', 6767, 3)
;

Я хочу получить PID последней строки для каждого рида (1 результат на уникальный RID). Для примера данных я бы хотел:

pid | MAX(timestamp)      | rid
-----------------------------------
5   | 2011-04-14 01:06:06 | 1
3   | 2011-04-14 01:05:41 | 2
7   | 2011-04-14 01:14:14 | 3

Я попытался выполнить следующий запрос:

SELECT MAX(timestamp),rid,pid FROM theTable GROUP BY rid

и я получаю:

max(timestamp)     ; rid; pid
----------------------------
2011-04-14 01:06:06; 1  ; 1
2011-04-14 01:05:41; 2  ; 3
2011-04-14 01:14:14; 3  ; 7

Возвращенный PID всегда является первым случаем PID для RID (строка / pid 1 используется впервые при освобождении 1, row / pid 3 при первом использовании RID 2, строка / pid 7 освобождается впервые 3 используется). Хотя возвращаются максимальные временные метки для каждого рида, pids не являются pids для временных меток из исходной таблицы. Какой запрос даст мне результаты, которые я ищу?

Ответы [ 7 ]

51 голосов
/ 14 апреля 2011

(протестировано в PostgreSQL 9. что-то)

Идентифицируйте рид и отметку времени.

select rid, max(timestamp) as ts
from test
group by rid;

1   2011-04-14 18:46:00
2   2011-04-14 14:59:00

Присоединитесь к нему.

select test.pid, test.cost, test.timestamp, test.rid
from test
inner join 
    (select rid, max(timestamp) as ts
    from test
    group by rid) maxt
on (test.rid = maxt.rid and test.timestamp = maxt.ts)
7 голосов
/ 27 апреля 2014
select *
from (
    select `pid`, `timestamp`, `cost`, `rid`
    from theTable 
    order by `timestamp` desc
) as mynewtable
group by mynewtable.`rid`
order by mynewtable.`timestamp`

Надеюсь, я помог!

4 голосов
/ 14 апреля 2011
SELECT t.pid, t.cost, to.timestamp, t.rid
FROM test as t
JOIN (
    SELECT rid, max(tempstamp) AS maxtimestamp
    FROM test GROUP BY rid
) AS tmax
    ON t.pid = tmax.pid and t.timestamp = tmax.maxtimestamp
2 голосов
/ 06 марта 2013

Я создал индекс для rid и timestamp.

SELECT test.pid, test.cost, test.timestamp, test.rid
FROM theTable AS test
LEFT JOIN theTable maxt 
ON maxt.rid = test.rid
AND maxt.timestamp > test.timestamp
WHERE maxt.rid IS NULL 

Отображение строк 0 - 2 (всего 3, запрос занял 0,0104 с)

Этот метод выберет все желаемые значения из theTable (тест), оставляя само соединение (maxt) на всех отметках времени выше, чем на тесте с таким же освобождением. Когда отметка времени уже является самой высокой в ​​тесте, в maxt нет совпадений - что мы и ищем - значения в maxt становятся NULL. Теперь мы используем предложение WHERE maxt.rid IS NULL или любой другой столбец в maxt.

0 голосов
/ 15 октября 2014

Если вы хотите избежать присоединения, вы можете использовать:

SELECT pid, rid FROM theTable t1 WHERE t1.pid IN ( SELECT MAX(t2.pid) FROM theTable t2 GROUP BY t2.rid);
0 голосов
/ 14 апреля 2011

У вас также могут быть такие подзапросы:

SELECT ( SELECT MIN(t2.pid)
         FROM test t2
         WHERE t2.rid = t.rid
           AND t2.timestamp = maxtimestamp
       ) AS pid 
     , MAX(t.timestamp) AS maxtimestamp
     , t.rid
FROM test t
GROUP BY t.rid

Но, таким образом, вам понадобится еще один подзапрос, если вы хотите, чтобы cost было включено в показанные столбцы и т. Д.

Итак, group by и join - лучшее решение.

0 голосов
/ 14 апреля 2011

Попробуйте:

select pid,cost, timestamp, rid from theTable order by timestamp DESC limit 2;
...