Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Deep Tech Revolution: così Area Science Park apre i laboratori alle startup
Deep Tech Revolution: così Area Science Park apre i laboratori alle startup
Siamo tornati nel parco tecnologico di Trieste per il kick-off del programma che mette a disposizione di cinque startup le infrastrutture di ricerca, dal sincrotrone Elettra ai laboratori di genomica e HPC. Roberto Pillon racconta il modello e la visione
HP OMEN MAX 16 con RTX 5080: potenza da desktop replacement a prezzo competitivo
HP OMEN MAX 16 con RTX 5080: potenza da desktop replacement a prezzo competitivo
HP OMEN MAX 16-ak0001nl combina RTX 5080 Laptop e Ryzen AI 9 HX 375 in un desktop replacement potente e ben raffreddato, con display 240 Hz e dotazione completa. Autonomia limitata e calibrazione non perfetta frenano l'entusiasmo, ma a 2.609 euro è tra le proposte più interessanti della categoria.
Recensione Google Pixel 10a, si migliora poco ma è sempre un'ottima scelta
Recensione Google Pixel 10a, si migliora poco ma è sempre un'ottima scelta
Google ha appena rinnovato la sua celebre serie A con il Pixel 10a, lo smartphone della serie più conveniente se consideriamo il rapporto tra costo e prestazioni. Con il chip Tensor G4, un design raffinato soprattutto sul retro e l'integrazione profonda di Gemini, il colosso di Mountain View promette un'esperienza premium a un prezzo accessibile. E il retro non ha nessuno scalino
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 26-06-2008, 10:09   #1
Argosoft
Senior Member
 
L'Avatar di Argosoft
 
Iscritto dal: Aug 2003
Città: addio fabriANO... mò sto a Roma ahò!!
Messaggi: 579
[mySQL] dubbio sempiterno sul GROUP BY

Della serie: ora basta!! è ora di finirla!!

Allora, devo togliermi questo dubbio definitivamente sennò non dormo più la notte.

Poniamo un esempio qualsiasi. Io ho una lista di persone in un'azienda:

Codice:
ID            Ruolo                    Nome

 1            Segreteria               Andrea Rossi
 2            Segreteria               Giovanni Rossi
 3            Amministrazione          Peppe Rossi
 4            Impiegato                Giulia Rossi
 5            Impiegato                Ubaldo Rossi
 6            Impiegato                Gianmarco Rossi
 7            Impiegato                Francesca Rossi
 8            Impiegato                Elena Rossi
 9            Impiegato                Carlo Rossi
10            Pianificazione           Paolo Rossi
11            Pulizie                  Simone Rossi
12            Pulizie                  Genoveffa Rossi

Se io facessi una select con group by sul Ruolo, avrei un risultato simile:


Codice:
ID            Ruolo                    Nome

 1            Segreteria               Andrea Rossi
 3            Amministrazione          Peppe Rossi
 4            Impiegato                Giulia Rossi
10            Pianificazione           Paolo Rossi
12            Pulizie                  Genoveffa Rossi
Però... con che criterio???
Voglio dire: ok, i dati vengono raggruppati per ruolo, ma quali righe per ogni gruppo vengono scelte? Le prime incontrate in ordine di presenza nel db? Le ultime? A random? E SOPRATTUTTO:
Posso scegliere quali righe farmi restituire?
E' un problema che ho incontrato un sacco di volte e non ne sono mai venuto a capo. Nell'esempio di prima, se io per ogni gruppo volessi prendere che ne so, l'ultimo elemento in ordine alfabetico? O il primo? O comunque qualcosa in particolare?

Ho guardato anche qui ma non ho avuto speranze lo stesso:
http://dev.mysql.com/tech-resources/...-by-myths.html


Qualcuno sa aiutarmi?! se mi risolvete questa.... se anche mi dite NON SE PO' FA' io torno a dormire la notte


Thanks!!
Argosoft è offline   Rispondi citando il messaggio o parte di esso
Old 26-06-2008, 10:51   #2
john_revelator
Senior Member
 
L'Avatar di john_revelator
 
Iscritto dal: Jul 2007
Messaggi: 1092
Ciao, vedi se questo link può esserti utile.

http://forum.html.it/forum/showthrea...t=group+by+max
john_revelator è offline   Rispondi citando il messaggio o parte di esso
Old 26-06-2008, 16:35   #3
Argosoft
Senior Member
 
L'Avatar di Argosoft
 
Iscritto dal: Aug 2003
Città: addio fabriANO... mò sto a Roma ahò!!
Messaggi: 579
Uhm..... quindi dici che nel singolo group by non c'è possibilità di scelta? Bisogna per forza usare una subquery?
Argosoft è offline   Rispondi citando il messaggio o parte di esso
Old 26-06-2008, 16:40   #4
john_revelator
Senior Member
 
L'Avatar di john_revelator
 
Iscritto dal: Jul 2007
Messaggi: 1092
Per quanto io sappia non è possibile con una semplice query estrarre il valore maggiore di ciascun gruppo o effettuare operazioni simili.
john_revelator è offline   Rispondi citando il messaggio o parte di esso
Old 26-06-2008, 17:28   #5
gugoXX
Senior Member
 
L'Avatar di gugoXX
 
Iscritto dal: May 2004
Città: Londra (Torino)
Messaggi: 3692
Quote:
Originariamente inviato da Argosoft Guarda i messaggi
Della serie: ora basta!! è ora di finirla!!

Allora, devo togliermi questo dubbio definitivamente sennò non dormo più la notte.

Poniamo un esempio qualsiasi. Io ho una lista di persone in un'azienda:

...

Se io facessi una select con group by sul Ruolo, avrei un risultato simile:


Codice:
ID            Ruolo                    Nome

 1            Segreteria               Andrea Rossi
 3            Amministrazione          Peppe Rossi
 4            Impiegato                Giulia Rossi
10            Pianificazione           Paolo Rossi
12            Pulizie                  Genoveffa Rossi
Però... con che criterio???
QUESTA QUERY e' sbagliata.
quasliasi motore di database serio impedisce l'esecuzioni di query come questa.
MySql evidentemente non e' serio.
Quando si opera con GROUP BY, si possono selezionare solo le colonne messe in GROUP BY e funzioni di gruppo (e costanti e funzioni scalari che accettano parametri scelti tra le colonne in group by)

Quote:
Voglio dire: ok, i dati vengono raggruppati per ruolo, ma quali righe per ogni gruppo vengono scelte? Le prime incontrate in ordine di presenza nel db? Le ultime? A random? E SOPRATTUTTO:
Posso scegliere quali righe farmi restituire?
esattamente per questo motivo i motori seri impediscono tali query, perche' il risultato e' impredicibile o arbitrariamente scelto dal motore secondo regole non standard.

Quindi... cosa vorresti fare? E troveremo una query.
__________________
Se pensi che il tuo codice sia troppo complesso da capire senza commenti, e' segno che molto probabilmente il tuo codice e' semplicemente mal scritto.
E se pensi di avere bisogno di un nuovo commento, significa che ti manca almeno un test.
gugoXX è offline   Rispondi citando il messaggio o parte di esso
Old 26-06-2008, 21:25   #6
kk3z
Senior Member
 
L'Avatar di kk3z
 
Iscritto dal: Nov 2003
Messaggi: 980
Non è vero che è sbagliata, è che non è questo il suo uso. Chi se ne frega di quale venga estratto, l'importante è che non vengono estratte tutte (o quella che uno vuole) e quindi non serve a questo. Ma se uno per esempio vuole contare quanti impiegati ci sono per ogni ruolo, perchè non la si dovrebbe usare?
kk3z è offline   Rispondi citando il messaggio o parte di esso
Old 26-06-2008, 21:33   #7
gugoXX
Senior Member
 
L'Avatar di gugoXX
 
Iscritto dal: May 2004
Città: Londra (Torino)
Messaggi: 3692
Quote:
Originariamente inviato da kk3z Guarda i messaggi
Non è vero che è sbagliata, è che non è questo il suo uso. Chi se ne frega di quale venga estratto, l'importante è che non vengono estratte tutte (o quella che uno vuole) e quindi non serve a questo. Ma se uno per esempio vuole contare quanti impiegati ci sono per ogni ruolo, perchè non la si dovrebbe usare?
Perche' la query per estrarre i dati di cui sopra e'
SELECT Id, Ruolo, Nome
FROM tabella
GROUP BY Id

e da SQL-92 standard (e successivi) non potresti estrarre Ruolo e Nome, dato che non fanno parte della GROUP BY.
E fra l'altro mi risulta possa essere eseguita solo con MySQL.

Per contare quanti impiegati ci sono in ogni ruolo la query e' un'altra
SELECT Ruolo, COUNT(*)
FROM tabella
GROUP BY Ruolo
__________________
Se pensi che il tuo codice sia troppo complesso da capire senza commenti, e' segno che molto probabilmente il tuo codice e' semplicemente mal scritto.
E se pensi di avere bisogno di un nuovo commento, significa che ti manca almeno un test.
gugoXX è offline   Rispondi citando il messaggio o parte di esso
Old 27-06-2008, 10:10   #8
Argosoft
Senior Member
 
L'Avatar di Argosoft
 
Iscritto dal: Aug 2003
Città: addio fabriANO... mò sto a Roma ahò!!
Messaggi: 579
Quote:
Originariamente inviato da gugoXX Guarda i messaggi
Perche' la query per estrarre i dati di cui sopra e'
SELECT Id, Ruolo, Nome
FROM tabella
GROUP BY Id

e da SQL-92 standard (e successivi) non potresti estrarre Ruolo e Nome, dato che non fanno parte della GROUP BY.
E fra l'altro mi risulta possa essere eseguita solo con MySQL.

Per contare quanti impiegati ci sono in ogni ruolo la query e' un'altra
SELECT Ruolo, COUNT(*)
FROM tabella
GROUP BY Ruolo
Questa non la sapevo, evidentemente mi sono abituato male?

Non è che io abbia un obiettivo in particolare, non cerco una query precisa, mi interessava appunto sapere come venivano estratti i risultati dentro a un group by, cosa che da quanto ho capito da questo passaggio

Quote:
esattamente per questo motivo i motori seri impediscono tali query, perche' il risultato e' impredicibile o arbitrariamente scelto dal motore secondo regole non standard.
non mi dà molte possibilità


Anzi no, un esempio ce l'ho.
Mettiamo il caso di una fotogallery, che ha delle categorie e delle foto. Con una query (senza subquery magari) vorrei tirare fuori la lista delle categorie con inclusa l'ultima foto inserita in ognuna (indicazione che prenderei dalla data di inserimento della foto).
Se potessi, farei un group by ordinando i risultati del gruppo delle foto e scegliendone appunto una in particolare, cioè la maggiore in data.

Tutto questo in una sola query. E' possibile?

Thanks!
Argosoft è offline   Rispondi citando il messaggio o parte di esso
Old 27-06-2008, 15:40   #9
john_revelator
Senior Member
 
L'Avatar di john_revelator
 
Iscritto dal: Jul 2007
Messaggi: 1092
Per fare ciò che chiedi è necessario l'uso delle subquery, per lo meno in mysql. Il link che ti ho indicato mi sembra faccia proprio al caso tuo.
john_revelator è offline   Rispondi citando il messaggio o parte di esso
Old 27-06-2008, 17:01   #10
gugoXX
Senior Member
 
L'Avatar di gugoXX
 
Iscritto dal: May 2004
Città: Londra (Torino)
Messaggi: 3692
Quote:
Originariamente inviato da Argosoft Guarda i messaggi
Anzi no, un esempio ce l'ho.
Mettiamo il caso di una fotogallery, che ha delle categorie e delle foto. Con una query (senza subquery magari) vorrei tirare fuori la lista delle categorie con inclusa l'ultima foto inserita in ognuna (indicazione che prenderei dalla data di inserimento della foto).
Se potessi, farei un group by ordinando i risultati del gruppo delle foto e scegliendone appunto una in particolare, cioè la maggiore in data.

Tutto questo in una sola query. E' possibile?

Thanks!
E' possibile in piu' modi. Il piu' semplice che mi viene in mente e' usando una INLINE query, che e' una SELECT che restitusice uno scalare, e che puo' dipendere da altre colonne di ciascun record.
Non conosco alcun motore SQL che non le permetta (access forse?)

Innanzitutto per risolvere la tua domanda devi essere SICURO a priori che la foto scattata per ultima sia una sola, ovvero che non ci siano 2 foto scattate dalla stessa persona nello stesso momento, pena ECCEZIONE durante l'esecuzione.
Se questo e' verificato sarebbe anche bene rinforzarlo a livello database con un constraint UNIQUE (user_id, datafoto)

Sarebbe qualcosa tipo:

Codice:
SELECT user_id,
           MAX(datafoto) as dataultimafoto,
           (SELECT codicefoto
                FROM tabella tint
                WHERE tint.user_id=text.user_id AND tint.datafoto=MAX(text.datafoto)
            ) AS codiceultimafoto
FROM tabella text
GROUP BY user_id
Dove al posto di codicefoto puoi mettere uno qualunque dei campi che ti servono dell'ultima foto (filename?)
La parte dentro la parentesi e' una INLINE query, che serve di fatto a calcolare il valore della colonna "codiceultimafoto" per ciascuna delle righe della master table, che per costrutto sara' una riga per ciascun utente.

Se i campi che ti servono per l'ultima di ciascuno fossero 2 o piu' di 2, la INLINE QUERY e' sconsigliata, dato che dovresti scrivere una costrutto analogo per ciascuna delle colonne interessanti.
Si procederebbe con una JOIN con una subquery, con un costrutto abbastanza diverso.

Codice:
SELECT * FROM (
   SELECT user_id,MAX(datafoto) dataultimafoto
   FROM tabella
   GROUP BY user_id
   ) AS master JOIN tabella detail
   ON (master.user_id=tabella.user_id
      AND master.dataultimafoto=tabella.datafoto)
Anche qui devi essere SICURO a priori che la foto scattata per ultima sia una sola, ovvero che non ci siano 2 foto scattate dalla stessa persona nello stesso momento, pena DUPLICAZIONE dati durante l'esecuzione.

Qualsiasi scelta tu faccia, accederai 2 volte alla tabella, una volta per cercare qual e' l'ultima foto, un'altra volta per andarne a pescare i dati.
Per questo motivo nell'SQL sono state introdotte le ANALYTIC function, che permettono di effettuare operazioni di correlazione intrariga.
Se MySql le supporta (ma non sono tanto standard, quasi tutti ce le hanno oggi, ma magari con nomi diversi)
puoi fare qualcosa analogo al seguente, codice per Oracle.

Codice:
SELECT DISTINCT user_id, 
           MAX(datafoto) OVER (PARTITION BY user_id) dataultimafoto,
           FIRST_VALUE(nomefile) OVER (PARTITION BY user_id ORDER BY datafoto desc) nomefileultimafoto,
           FIRST_VALUE(attributoX) OVER (PARTITION BY user_id ORDER BY datafoto desc) attributoXultimafoto
FROM tabella
Dove la tabella viene letta e smastruzzata una volta sola, e dove puoi avere tutti gli attributi che vuoi relativamente all'ultima foto.
E dove non devi neppure avere il constraint di unicita' della data dell'ultima foto. In questo caso la query funzionera', ma dove a parita' di ultima data ti restituierebbe uno a caso degli attributi di una delle N ultime foto (non necessariamente appartenenti tutti alla stessa delle N foto).

Ultimo caso, che serve precisamente a risolvere questo problema, ma dove ho trovato solo Oracle in grado di risolverlo, e' quello di usare funzioni analitiche di gruppo.

Codice:
SELECT user_id, 
           MAX(datafoto) dataultimafoto,
           MAX(nomefile) KEEP (DENSE_RANK LAST ORDER BY datafoto) nomefileultimafoto,
           MAX(attributoX) KEEP (DENSE_RANK LAST ORDER BY datafoto) attributoXultimafoto
FROM tabella
GROUP BY user_id
La cui spiegazione aprirei solo agli interessati.
e che in Oracle darebbe direi il piano di esecuzione migliore.
__________________
Se pensi che il tuo codice sia troppo complesso da capire senza commenti, e' segno che molto probabilmente il tuo codice e' semplicemente mal scritto.
E se pensi di avere bisogno di un nuovo commento, significa che ti manca almeno un test.
gugoXX è offline   Rispondi citando il messaggio o parte di esso
Old 27-06-2008, 17:33   #11
john_revelator
Senior Member
 
L'Avatar di john_revelator
 
Iscritto dal: Jul 2007
Messaggi: 1092
@gugoxx: leggo sempre con molto interesse le tue dettagliate spiegazioni sull'SQL. Ciao e complimenti.
john_revelator è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Deep Tech Revolution: così Area Science Park apre i laboratori alle startup Deep Tech Revolution: così Area Science P...
HP OMEN MAX 16 con RTX 5080: potenza da desktop replacement a prezzo competitivo HP OMEN MAX 16 con RTX 5080: potenza da desktop ...
Recensione Google Pixel 10a, si migliora poco ma è sempre un'ottima scelta Recensione Google Pixel 10a, si migliora poco ma...
6G, da rete che trasporta dati a rete intelligente: Qualcomm accelera al MWC 2026 6G, da rete che trasporta dati a rete intelligen...
CHUWI CoreBook Air alla prova: design premium, buona autonomia e qualche compromesso CHUWI CoreBook Air alla prova: design premium, b...
Il nuovo MacBook Neo ha una memoria SSD ...
Xbox Project Helix, le prime specifiche ...
Annunci pubblicitari sulla TV quando cam...
Prezzi aumentati del 50% durante la nott...
Sconti studiati per singolo utente: Sony...
Addio alla Kia Niro EV, il crossover sar...
Apple crede nel suo iPhone Fold: la prod...
Fortnite, un nuovo listino per i pacchet...
Ecco i nuovi Sonos Play ed Era 100 SL: d...
Razer svela il futuro del gaming potenzi...
Tre robot Narwal in offerta: pulizia aut...
Gracenote denuncia OpenAI: ChatGPT addes...
Microsoft AI Tour Milano: dall'efficienz...
Asus ExpertBook Ultra: Intel Core Ultra ...
Intel presenta i processori desktop Core...
Chromium
GPU-Z
OCCT
LibreOffice Portable
Opera One Portable
Opera One 106
CCleaner Portable
CCleaner Standard
Cpu-Z
Driver NVIDIA GeForce 546.65 WHQL
SmartFTP
Trillian
Google Chrome Portable
Google Chrome 120
VirtualBox
Tutti gli articoli Tutte le news Tutti i download

Strumenti

Regole
Non Puoi aprire nuove discussioni
Non Puoi rispondere ai messaggi
Non Puoi allegare file
Non Puoi modificare i tuoi messaggi

Il codice vB è On
Le Faccine sono On
Il codice [IMG] è On
Il codice HTML è Off
Vai al Forum


Tutti gli orari sono GMT +1. Ora sono le: 01:11.


Powered by vBulletin® Version 3.6.4
Copyright ©2000 - 2026, Jelsoft Enterprises Ltd.
Served by www3v