Come posso SELEZIONARE le righe con MAX (valore colonna), DISTINCT di un'altra colonna in SQL?
-
03-07-2019 - |
Domanda
La mia tabella è:
id home datetime player resource
---|-----|------------|--------|---------
1 | 10 | 04/03/2009 | john | 399
2 | 11 | 04/03/2009 | juliet | 244
5 | 12 | 04/03/2009 | borat | 555
3 | 10 | 03/03/2009 | john | 300
4 | 11 | 03/03/2009 | juliet | 200
6 | 12 | 03/03/2009 | borat | 500
7 | 13 | 24/12/2008 | borat | 600
8 | 13 | 01/01/2009 | borat | 700
Devo selezionare ogni home
distinto che contenga il valore massimo di datetime
.
Il risultato sarebbe:
id home datetime player resource
---|-----|------------|--------|---------
1 | 10 | 04/03/2009 | john | 399
2 | 11 | 04/03/2009 | juliet | 244
5 | 12 | 04/03/2009 | borat | 555
8 | 13 | 01/01/2009 | borat | 700
Ho provato:
-- 1 ..by the MySQL manual:
SELECT DISTINCT
home,
id,
datetime AS dt,
player,
resource
FROM topten t1
WHERE datetime = (SELECT
MAX(t2.datetime)
FROM topten t2
GROUP BY home)
GROUP BY datetime
ORDER BY datetime DESC
Non funziona. Il set di risultati ha 130 righe sebbene il database ne contenga 187.
Il risultato include alcuni duplicati di home
.
-- 2 ..join
SELECT
s1.id,
s1.home,
s1.datetime,
s1.player,
s1.resource
FROM topten s1
JOIN (SELECT
id,
MAX(datetime) AS dt
FROM topten
GROUP BY id) AS s2
ON s1.id = s2.id
ORDER BY datetime
No. Fornisce tutti i record.
-- 3 ..something exotic:
Con vari risultati.
Soluzione
Sei così vicino! Tutto quello che devi fare è selezionare ENTRAMBI la home e la sua data massima, quindi tornare alla tabella topten
su ENTRAMBI i campi:
SELECT tt.*
FROM topten tt
INNER JOIN
(SELECT home, MAX(datetime) AS MaxDateTime
FROM topten
GROUP BY home) groupedtt
ON tt.home = groupedtt.home
AND tt.datetime = groupedtt.MaxDateTime
Altri suggerimenti
Ecco la versione T-SQL :
-- Test data
DECLARE @TestTable TABLE (id INT, home INT, date DATETIME,
player VARCHAR(20), resource INT)
INSERT INTO @TestTable
SELECT 1, 10, '2009-03-04', 'john', 399 UNION
SELECT 2, 11, '2009-03-04', 'juliet', 244 UNION
SELECT 5, 12, '2009-03-04', 'borat', 555 UNION
SELECT 3, 10, '2009-03-03', 'john', 300 UNION
SELECT 4, 11, '2009-03-03', 'juliet', 200 UNION
SELECT 6, 12, '2009-03-03', 'borat', 500 UNION
SELECT 7, 13, '2008-12-24', 'borat', 600 UNION
SELECT 8, 13, '2009-01-01', 'borat', 700
-- Answer
SELECT id, home, date, player, resource
FROM (SELECT id, home, date, player, resource,
RANK() OVER (PARTITION BY home ORDER BY date DESC) N
FROM @TestTable
)M WHERE N = 1
-- and if you really want only home with max date
SELECT T.id, T.home, T.date, T.player, T.resource
FROM @TestTable T
INNER JOIN
( SELECT TI.id, TI.home, TI.date,
RANK() OVER (PARTITION BY TI.home ORDER BY TI.date) N
FROM @TestTable TI
WHERE TI.date IN (SELECT MAX(TM.date) FROM @TestTable TM)
)TJ ON TJ.N = 1 AND T.id = TJ.id
Modifica
Sfortunatamente, non ci sono funzioni RANK () OVER in MySQL.
Ma può essere emulato, vedi Funzioni di emulazione analitica (ranking AKA) con MySQL .
Quindi questa è la versione MySQL :
SELECT id, home, date, player, resource
FROM TestTable AS t1
WHERE
(SELECT COUNT(*)
FROM TestTable AS t2
WHERE t2.home = t1.home AND t2.date > t1.date
) = 0
La soluzione MySQL
più veloce, senza query interne e senza GROUP BY
:
SELECT m.* -- get the row that contains the max value
FROM topten m -- "m" from "max"
LEFT JOIN topten b -- "b" from "bigger"
ON m.home = b.home -- match "max" row with "bigger" row by `home`
AND m.datetime < b.datetime -- want "bigger" than "max"
WHERE b.datetime IS NULL -- keep only if there is no bigger than max
Spiegazione :
Unisciti alla tabella con se stesso usando la colonna home
. L'uso di LEFT JOIN
garantisce che tutte le righe della tabella m
compaiano nel set di risultati. Quelli che non hanno una corrispondenza nella tabella b
avranno NULL
s per le colonne di b
.
L'altra condizione su JOIN
richiede di abbinare solo le righe da b
che hanno un valore maggiore nella colonna datetime
rispetto alla riga da m
.
Utilizzando i dati pubblicati nella domanda, LEFT JOIN
produrrà queste coppie:
+------------------------------------------+--------------------------------+
| the row from `m` | the matching row from `b` |
|------------------------------------------|--------------------------------|
| id home datetime player resource | id home datetime ... |
|----|-----|------------|--------|---------|------|------|------------|-----|
| 1 | 10 | 04/03/2009 | john | 399 | NULL | NULL | NULL | ... | *
| 2 | 11 | 04/03/2009 | juliet | 244 | NULL | NULL | NULL | ... | *
| 5 | 12 | 04/03/2009 | borat | 555 | NULL | NULL | NULL | ... | *
| 3 | 10 | 03/03/2009 | john | 300 | 1 | 10 | 04/03/2009 | ... |
| 4 | 11 | 03/03/2009 | juliet | 200 | 2 | 11 | 04/03/2009 | ... |
| 6 | 12 | 03/03/2009 | borat | 500 | 5 | 12 | 04/03/2009 | ... |
| 7 | 13 | 24/12/2008 | borat | 600 | 8 | 13 | 01/01/2009 | ... |
| 8 | 13 | 01/01/2009 | borat | 700 | NULL | NULL | NULL | ... | *
+------------------------------------------+--------------------------------+
Infine, la clausola WHERE
mantiene solo le coppie che hanno NULL
nelle colonne di b
(sono contrassegnate con *
nella tabella sopra); ciò significa che, a causa della seconda condizione dalla clausola JOIN
, la riga selezionata da m
ha il valore più grande nella colonna datetime
.
Leggi il SQL Antipatterns: evitare le insidie ??della programmazione di database per altri suggerimenti SQL .
Funzionerà anche se hai due o più righe per ogni home
con DATETIME
uguali:
SELECT id, home, datetime, player, resource
FROM (
SELECT (
SELECT id
FROM topten ti
WHERE ti.home = t1.home
ORDER BY
ti.datetime DESC
LIMIT 1
) lid
FROM (
SELECT DISTINCT home
FROM topten
) t1
) ro, topten t2
WHERE t2.id = ro.lid
Penso che questo ti darà il risultato desiderato:
SELECT home, MAX(datetime)
FROM my_table
GROUP BY home
MA se hai bisogno anche di altre colonne, devi solo unirti alla tabella originale (controlla Michael La Voie
rispondi)
Cordiali saluti.
Dal momento che le persone sembrano continuare a imbattersi in questa discussione (la data dei commenti varia da 1,5 anni) non è molto più semplice:
SELEZIONA * DA (SELEZIONA * DA topten ORDER BY datetime DESC) tmp GROUP BY home
Non sono necessarie funzioni di aggregazione ...
Saluti.
Puoi anche provare questo e per le tabelle di grandi dimensioni le prestazioni delle query saranno migliori. Funziona quando non ci sono più di due record per ogni casa e le loro date sono diverse. Una query MySQL generale migliore è quella di Michael La Voie sopra.
SELECT t1.id, t1.home, t1.date, t1.player, t1.resource
FROM t_scores_1 t1
INNER JOIN t_scores_1 t2
ON t1.home = t2.home
WHERE t1.date > t2.date
O in caso di Postgres o quei dbs che forniscono funzioni analitiche, prova
SELECT t.* FROM
(SELECT t1.id, t1.home, t1.date, t1.player, t1.resource
, row_number() over (partition by t1.home order by t1.date desc) rw
FROM topten t1
INNER JOIN topten t2
ON t1.home = t2.home
WHERE t1.date > t2.date
) t
WHERE t.rw = 1
Funziona su Oracle:
with table_max as(
select id
, home
, datetime
, player
, resource
, max(home) over (partition by home) maxhome
from table
)
select id
, home
, datetime
, player
, resource
from table_max
where home = maxhome
SELECT tt.*
FROM TestTable tt
INNER JOIN
(
SELECT coord, MAX(datetime) AS MaxDateTime
FROM rapsa
GROUP BY
krd
) groupedtt
ON tt.coord = groupedtt.coord
AND tt.datetime = groupedtt.MaxDateTime
Prova questo per SQL Server:
WITH cte AS (
SELECT home, MAX(year) AS year FROM Table1 GROUP BY home
)
SELECT * FROM Table1 a INNER JOIN cte ON a.home = cte.home AND a.year = cte.year
SELECT c1, c2, c3, c4, c5 FROM table1 WHERE c3 = (select max(c3) from table)
SELECT * FROM table1 WHERE c3 = (select max(c3) from table1)
Ecco la versione di MySQL che stampa solo una voce in cui ci sono duplicati MAX (datetime) in un gruppo.
Puoi testare qui http://www.sqlfiddle.com/#!2/ 0a4ae / 1
Dati di esempio
mysql> SELECT * from topten;
+------+------+---------------------+--------+----------+
| id | home | datetime | player | resource |
+------+------+---------------------+--------+----------+
| 1 | 10 | 2009-04-03 00:00:00 | john | 399 |
| 2 | 11 | 2009-04-03 00:00:00 | juliet | 244 |
| 3 | 10 | 2009-03-03 00:00:00 | john | 300 |
| 4 | 11 | 2009-03-03 00:00:00 | juliet | 200 |
| 5 | 12 | 2009-04-03 00:00:00 | borat | 555 |
| 6 | 12 | 2009-03-03 00:00:00 | borat | 500 |
| 7 | 13 | 2008-12-24 00:00:00 | borat | 600 |
| 8 | 13 | 2009-01-01 00:00:00 | borat | 700 |
| 9 | 10 | 2009-04-03 00:00:00 | borat | 700 |
| 10 | 11 | 2009-04-03 00:00:00 | borat | 700 |
| 12 | 12 | 2009-04-03 00:00:00 | borat | 700 |
+------+------+---------------------+--------+----------+
Versione MySQL con variabile utente
SELECT *
FROM (
SELECT ord.*,
IF (@prev_home = ord.home, 0, 1) AS is_first_appear,
@prev_home := ord.home
FROM (
SELECT t1.id, t1.home, t1.player, t1.resource
FROM topten t1
INNER JOIN (
SELECT home, MAX(datetime) AS mx_dt
FROM topten
GROUP BY home
) x ON t1.home = x.home AND t1.datetime = x.mx_dt
ORDER BY home
) ord, (SELECT @prev_home := 0, @seq := 0) init
) y
WHERE is_first_appear = 1;
+------+------+--------+----------+-----------------+------------------------+
| id | home | player | resource | is_first_appear | @prev_home := ord.home |
+------+------+--------+----------+-----------------+------------------------+
| 9 | 10 | borat | 700 | 1 | 10 |
| 10 | 11 | borat | 700 | 1 | 11 |
| 12 | 12 | borat | 700 | 1 | 12 |
| 8 | 13 | borat | 700 | 1 | 13 |
+------+------+--------+----------+-----------------+------------------------+
4 rows in set (0.00 sec)
Ritaglio delle risposte accettate
SELECT tt.*
FROM topten tt
INNER JOIN
(
SELECT home, MAX(datetime) AS MaxDateTime
FROM topten
GROUP BY home
) groupedtt ON tt.home = groupedtt.home AND tt.datetime = groupedtt.MaxDateTime
+------+------+---------------------+--------+----------+
| id | home | datetime | player | resource |
+------+------+---------------------+--------+----------+
| 1 | 10 | 2009-04-03 00:00:00 | john | 399 |
| 2 | 11 | 2009-04-03 00:00:00 | juliet | 244 |
| 5 | 12 | 2009-04-03 00:00:00 | borat | 555 |
| 8 | 13 | 2009-01-01 00:00:00 | borat | 700 |
| 9 | 10 | 2009-04-03 00:00:00 | borat | 700 |
| 10 | 11 | 2009-04-03 00:00:00 | borat | 700 |
| 12 | 12 | 2009-04-03 00:00:00 | borat | 700 |
+------+------+---------------------+--------+----------+
7 rows in set (0.00 sec)
Un altro modo per ottenere la riga più recente per gruppo utilizzando una query secondaria che in sostanza calcola un ranking per ogni riga per gruppo e quindi filtra le righe più recenti come con rank = 1
select a.*
from topten a
where (
select count(*)
from topten b
where a.home = b.home
and a.`datetime` < b.`datetime`
) +1 = 1
Ecco la demo visiva per il rango no per ogni riga per una migliore comprensione
Leggendo alcuni commenti che ne dici se ci sono due righe che hanno gli stessi valori dei campi "home" e "datetime"?
La query precedente non riuscirà e restituirà più di 1 riga per la situazione precedente. Per coprire questa situazione ci sarà bisogno di un altro criterio / parametro / colonna per decidere quale riga dovrebbe essere presa quale rientra nella situazione sopra. Osservando il set di dati di esempio, suppongo che sia presente una colonna chiave primaria id
che dovrebbe essere impostata su incremento automatico. Quindi possiamo usare questa colonna per selezionare la riga più recente modificando la stessa query con l'aiuto dell'istruzione CASE
come
select a.*
from topten a
where (
select count(*)
from topten b
where a.home = b.home
and case
when a.`datetime` = b.`datetime`
then a.id < b.id
else a.`datetime` < b.`datetime`
end
) + 1 = 1
La query sopra selezionerà la riga con l'id più alto tra gli stessi valori datetime
demo visiva per il rango no per ogni riga
Prova questo
select * from mytable a join
(select home, max(datetime) datetime
from mytable
group by home) b
on a.home = b.home and a.datetime = b.datetime
Saluti K
Perché non usare: SELEZIONA home, MAX (datetime) AS MaxDateTime, player, risorsa DA apt. GROUP BY home Mi sono perso qualcosa?
questa è la query di cui hai bisogno:
SELECT b.id, a.home,b.[datetime],b.player,a.resource FROM
(SELECT home,MAX(resource) AS resource FROM tbl_1 GROUP BY home) AS a
LEFT JOIN
(SELECT id,home,[datetime],player,resource FROM tbl_1) AS b
ON a.resource = b.resource WHERE a.home =b.home;
@Michae La risposta accettata funzionerà bene nella maggior parte dei casi, ma fallisce per uno come sotto.
Nel caso in cui vi fossero 2 righe con HomeID e Datetime uguali, la query restituirà entrambe le righe, non HomeID distinto come richiesto, per cui aggiungere Distinct nella query come di seguito.
SELECT DISTINCT tt.home , tt.MaxDateTime
FROM topten tt
INNER JOIN
(SELECT home, MAX(datetime) AS MaxDateTime
FROM topten
GROUP BY home) groupedtt
ON tt.home = groupedtt.home
AND tt.datetime = groupedtt.MaxDateTime