Delphi-PRAXiS

Delphi-PRAXiS (https://www.delphipraxis.net/forum.php)
-   Datenbanken (https://www.delphipraxis.net/15-datenbanken/)
-   -   HAVING mit MAX (alias) (https://www.delphipraxis.net/169256-having-mit-max-alias.html)

MrSpock 6. Jul 2012 19:05

Datenbank: MYSQL • Version: 5.1 • Zugriff über: Konsole

HAVING mit MAX (alias)
 
Hallo,

warum funktioniert folgende query nicht?

Code:
SELECT id, count(wert) as anz
GROUP BY id
HAVING (anz = MAX(anz))
Idee ist, dass ich die Einträge finde, die in der Spalte anz (alias für count(wert)) den maximalen Wert haben.

Interessanterweise werden alle Werte gezeigt, wenn ich schreibe

Code:
HAVING (Max(anz) = 0)
anz hat aber für alle Datensätze einen Wert > 0 !?

Bummi 6. Jul 2012 19:14

AW: HAVING mit MAX (alias)
 
Keine Ahnung wie weit MySQL inzwischen mit Subselects ist, geht sowas?
Code:
HAVING (Count(*) = (Select MAX(ANZ) from (SELECT id, count(wert) as anz
from ??
GROUP BY id) ))

Sir Rufo 6. Jul 2012 19:29

AW: HAVING mit MAX (alias)
 
So sollte das eigentlich funktionieren
SQL-Code:
SELECT * FROM (
  SELECT id, count(wert) as anz
  GROUP BY id
) as tmp
WHERE anz = MAX( anz );
aber das geht auch:
SQL-Code:
SELECT * FROM (
  SELECT id, count(wert) as anz
  GROUP BY id
) as tmp
ORDER BY anz DESC
LIMIT 1;

MrSpock 6. Jul 2012 19:39

AW: HAVING mit MAX (alias)
 
Hallo Sir Rufo,

so hab ich es auch schon versucht. Das Problem ist aber, dass mir immer nur ein Datensatz ausgegeben wird, obwohl mehrere den maximalen Wert haben.

bei anz = Max(anz) muss die HAVING Clause benutzt werden, weil where keine Aggregatfunktionen erlaubt

Namenloser 6. Jul 2012 20:09

AW: HAVING mit MAX (alias)
 
Warum es nicht funktioniert, kann ich dir auch nicht sagen, aber ich habe mal eben ein paar Varianten durchprobiert, und folgendes scheint zu funktionieren:

SQL-Code:
select * from Tabelle join (select max(Spalte) as maximum from Tabelle) as tmp on maximum=Spalte

MrSpock 6. Jul 2012 21:10

AW: HAVING mit MAX (alias)
 
Hallo Philip,

ich benötige ja das Maximum einer (alias Spalte), die wiederum eine Aggregatfunktion nutzt.

nahpets 6. Jul 2012 21:52

AW: HAVING mit MAX (alias)
 
Hallo,

mir ist es nur auf diese Art gelungen (allerdings kein MySQL):
Code:
select id, anz from (
  select id, count(wert) as anz
  from tabelle
  group by id
) a
where anz = (
  select max(anz) from (
    select id, count(wert) as anz
    from tabelle
    group by id
  )
)
Mit Having scheint es nicht unbedingt zu gehe(?)

Namenloser 6. Jul 2012 22:02

AW: HAVING mit MAX (alias)
 
Zitat:

Zitat von MrSpock (Beitrag 1173860)
Hallo Philip,

ich benötige ja das Maximum einer (alias Spalte), die wiederum eine Aggregatfunktion nutzt.

Ah sorry, hatte ich irgendwie falsch gesehn.

Ich habe folgende Varianten anzubieten:
SQL-Code:
select
  id,
  count(*) as anz,
  (
    select max(anz)
    from (select count(*) as anz from Tabelle group by id) as tmp
  ) as maximum
from Tabelle
group by id
having anz=maximum
SQL-Code:
select id, count(*) as anz
from Tabelle
group by id
having
  (
    select max(anz)
    from (select count(*) as anz from Tabelle group by id) as tmp
  ) = anz

Bummi 6. Jul 2012 22:56

AW: HAVING mit MAX (alias)
 
die von mir ursprünglich vorgeschlagene Lösung
Code:
SELECT id, count(*) as anz
from test
GROUP BY id
HAVING Count(*) = (Select MAX(ANZ) from (SELECT id, count(*) as anz
from test
GROUP BY id) i)
funktioniert unter MSSQL.
Funktioniert sie unter MySQL nicht?

Furtbichler 7. Jul 2012 07:25

AW: HAVING mit MAX (alias)
 
Zitat:

Zitat von MrSpock (Beitrag 1173849)
warum funktioniert folgende query nicht?

Weil die Syntax falsch ist: Du wendest in der HAVING-Klausel ein Aggregat auf eine Spalte der Ergebnistabelle an. Erlaubt sind jedoch nur Aggregate auf Spalten der Ursprungstabelle.
Zitat:

Interessanterweise werden alle Werte gezeigt, wenn ich schreibe
Code:
HAVING (Max(anz) = 0)

Ich finde das nicht interessant, sondern typisch mySQL.

Der Weg führt imho nur über eine 'temporäre Tabelle', d.h. die Aggregat-Tabelle mit den Spalten id und Anz (also Count(*)). Davon möchstest Du die Einträge, deren Wert Anz am größten ist.

Bei meiner Arbeit mit MSSQL ist mir aufgefallen, das die Verwendung temporärer Tabellen häufiger wesentlich performanter ist, leichter verständlich und schneller zum Ziel führt:
SQL-Code:
select Id, Count(*) as Anz into #temp from myTable group by Id
select * from #temp where Anz = (select max(Anz) from #temp)
Kennt mySQL temporäre Tabellen oder Tabellenvariablen? ....*gugel* ja, geht
SQL-Code:
create Temporary table Anzahl (id int, Anz int)
insert into Anzahl select id, count(*) from myTable
select * from Anzahl where Anz = (select max(Anz) from Anzahl)
DROP TABLE Anzahl
Hier scheint es so zu sein, das eine Lösung mit einem Statement nur über den Klimmzug geht, die Aggregattabelle 'Select id, count(*)...' mehrfach zu verwenden. Schafft der Compiler es, dies zu erkennen und das Aggregat nur 1x durchzuführen?

MrSpock 7. Jul 2012 15:06

AW: HAVING mit MAX (alias)
 
Zitat:

Zitat von Bummi (Beitrag 1173864)
die von mir ursprünglich vorgeschlagene Lösung
Code:
SELECT id, count(*) as anz
from test
GROUP BY id
HAVING Count(*) = (Select MAX(ANZ) from (SELECT id, count(*) as anz
from test
GROUP BY id) i)
funktioniert unter MSSQL.
Funktioniert sie unter MySQL nicht?

Hallo Bummi,
die Abfrage ist Wirklichkeit deutlich komplexer, deshalb war es mit dem Testen nicht so einfach. Ich habe es aber nochmal ausprobiert und behaupte, dass es nicht funktioniert!
Es wird mir kein Datensatz ausgegeben. Ich vermute, dass die Gleichung in der Having Clause nicht funktioniert.

Zitat:

Zitat von Furtbichler (Beitrag 1173878)
Zitat:

Zitat von MrSpock (Beitrag 1173849)
warum funktioniert folgende query nicht?

Weil die Syntax falsch ist: Du wendest in der HAVING-Klausel ein Aggregat auf eine Spalte der Ergebnistabelle an. Erlaubt sind jedoch nur Aggregate auf Spalten der Ursprungstabelle.
Zitat:

Interessanterweise werden alle Werte gezeigt, wenn ich schreibe
Code:
HAVING (Max(anz) = 0)

Ich finde das nicht interessant, sondern typisch mySQL.

Der Weg führt imho nur über eine 'temporäre Tabelle', d.h. die Aggregat-Tabelle mit den Spalten id und Anz (also Count(*)). Davon möchstest Du die Einträge, deren Wert Anz am größten ist.

Bei meiner Arbeit mit MSSQL ist mir aufgefallen, das die Verwendung temporärer Tabellen häufiger wesentlich performanter ist, leichter verständlich und schneller zum Ziel führt:
SQL-Code:
select Id, Count(*) as Anz into #temp from myTable group by Id
select * from #temp where Anz = (select max(Anz) from #temp)
Kennt mySQL temporäre Tabellen oder Tabellenvariablen? ....*gugel* ja, geht
SQL-Code:
create Temporary table Anzahl (id int, Anz int)
insert into Anzahl select id, count(*) from myTable
select * from Anzahl where Anz = (select max(Anz) from Anzahl)
DROP TABLE Anzahl
Hier scheint es so zu sein, das eine Lösung mit einem Statement nur über den Klimmzug geht, die Aggregattabelle 'Select id, count(*)...' mehrfach zu verwenden. Schafft der Compiler es, dies zu erkennen und das Aggregat nur 1x durchzuführen?

OK, diese Lösung scheint mir wirklich die beste zu sein. Ein Problem hat MySQL aber noch mit dem

SQL-Code:
select * from #temp where Anz = (select max(Anz) from #temp)
in einer Abfrage kann eine temporary table offensichtlich nicht zweimal benutzt werden. Deshalb musste ich noch eine zweite temp table benutzen :-(

Furtbichler 7. Jul 2012 17:14

AW: HAVING mit MAX (alias)
 
Hi Spock

Ich habe 0 Ahnung von mysql und deshalb sind meine Vorschläge nur Theorie.
Wenn mySQL mit Variablen umgehen kann, könntest Du dir die höchste Anzahl in einer Variablen merken und dann auf die 2.Tabelle verzichten.
So z.B.
SQL-Code:
DECLARE maxAnzahl INT
create Temporary table Anzahl (id int, Anz int)
insert into Anzahl select id, count(*) from myTable
select max(Anz) from Anzahl into maxAnzahl
select * from Anzahl where Anz = @maxAnzahl
DROP TABLE Anzahl
Wäre denkbar, das das nur in einer Prozedur oder einem BEGIN-END Block geht.

Namenloser 7. Jul 2012 18:00

AW: HAVING mit MAX (alias)
 
Die Lösung von Bummi und mir macht letztlich ja auch nichts anderes, nur dass die temporäre Tabelle nicht explizit angelegt wird.
SQL-Code:
select max(anz) from (
  select count(*) as anz from Tabelle group by id /* ← temporäre Tabelle */
)

Furtbichler 7. Jul 2012 19:34

AW: HAVING mit MAX (alias)
 
Na doch. Mr.Spock möchte alle Einträge, deren Spalte 'ID' am häufigsten vorkommt.
Deine Query liefert nur die maximale Anzahl.

Der kleine Frickelansatz von mir liefert zumindest die Spalten 'ID'

Namenloser 8. Jul 2012 00:00

AW: HAVING mit MAX (alias)
 
Zitat:

Zitat von Furtbichler (Beitrag 1173922)
Na doch. Mr.Spock möchte alle Einträge, deren Spalte 'ID' am häufigsten vorkommt.
Deine Query liefert nur die maximale Anzahl.

Nö.

Valle 8. Jul 2012 08:43

AW: HAVING mit MAX (alias)
 
Was ist damit?

[SQL]select * from test where value = (select max(value) from test);[/SQL]

Hat in meinen Tests funktioniert und ist irgendwie das einfachste?

Liebe Grüße,
Valentinf

Furtbichler 8. Jul 2012 10:32

AW: HAVING mit MAX (alias)
 
Zitat:

Zitat von NamenLozer (Beitrag 1173925)
Zitat:

Zitat von Furtbichler (Beitrag 1173922)
Na doch. Mr.Spock möchte alle Einträge, deren Spalte 'ID' am häufigsten vorkommt.
Deine Query liefert nur die maximale Anzahl.

Nö.

Wir reden aneinander vorbei: Mein 'Na doch' bezieht sich auf deinen letztgenannten Vorschlag:
Zitat:

Zitat von NamenLozer (Beitrag 1173918)
Die Lösung von Bummi und mir macht letztlich ja auch nichts anderes, nur dass die temporäre Tabelle nicht explizit angelegt wird.
SQL-Code:
select max(anz) from (
  select count(*) as anz from Tabelle group by id /* ← temporäre Tabelle */
)

Und das ist etwas komplett anderes, als das, was gefordert wurde.

Die von Dir zuvor präsentierten Lösungen (auf die Du dich im 'Nö' beziehst) funktionieren ja, aggregieren aber zweimal. Und ich habe eine Alternative vorgeschlagen, in der 'GROUP BY' nur 1x vorkommt.

tgvoelker 8. Jul 2012 11:58

AW: HAVING mit MAX (alias)
 
Code:
SELECT id, count(wert) as anz
GROUP BY id
HAVING (anz = MAX(anz))
Das kann nicht funktionieren, weil Du ein einfaches Aggregat eines Wertes mit einem mehrfachen Aggregat desselben vergleichen willst - was innerhalb einer SQL-Ebene (deren Aggregationsregeln Du mit GROUP BY definierst) unzulässig ist. Du erkennst das daran, daß MAX(ANZ) gruppenübergreifend ermittelt werden soll.

Where schränkt übrigens die ursprünglichen Datenmengen ein, d.h. wirkt vor der Gruppierung, kann deshalb die Aggregate nicht kennen.

Die folgende Syntax funktioniert (auf mySQL 5.1 getestet)

Code:
select
FileExt
,count(FileId) as Anz
from
filetable
group by FileExt
having
Anz=(select max(subanz) from (select count(FileId) as subanz from filetable group by FileExt)as subtab)
"FileExt" wäre Dein "ID", "FileId" entspräche Deinem "wert".

Namenloser 8. Jul 2012 19:05

AW: HAVING mit MAX (alias)
 
Zitat:

Zitat von Furtbichler (Beitrag 1173940)
Zitat:

SQL-Code:
select max(anz) from (
  select count(*) as anz from Tabelle group by id /* ← temporäre Tabelle */
)

Und das ist etwas komplett anderes, als das, was gefordert wurde.

Die von Dir zuvor präsentierten Lösungen (auf die Du dich im 'Nö' beziehst) funktionieren ja

Schon mal was von „Auszug“ gehört? :roll:

Furtbichler 8. Jul 2012 23:02

AW: HAVING mit MAX (alias)
 
Zitat:

Zitat von NamenLozer (Beitrag 1173980)
Schon mal was von „Auszug“ gehört? :roll:

Ja :roll:, was für eine blöde :roll: Frage. :roll:

Deine funktionierenden Vorschläge benötigen zwei GROUP BY Durchgänge, dein 'Auszug' kommt zwar nur mit einem GROUP BY aus, hat nichts mit der Lösung zu tun. Aber ich wiederhole mich.

Du hast verloren, gib einfach auf. :-D

Namenloser 8. Jul 2012 23:19

AW: HAVING mit MAX (alias)
 
Mag sein, dass deine Lösung einen Tick effizienter ist (falls MySQL nicht intelligent genug ist, das Zwischenergebnis zu cachen), aber darum ging es gar nicht. Beide Lösungen funktionieren, und zwar grundsätzlich auf die gleiche Weise; nichts anderes wollte ich mit dem Auszug zeigen. Es ist schon ein bisschen seltsam von dir, an einem Lösungsausschnitt zu monieren, dass er ohne den Rest, der zuvor ja vollständig gepostet wurde, nicht funktioniert :lol:

Aber genug damit.


Alle Zeitangaben in WEZ +1. Es ist jetzt 14:01 Uhr.

Powered by vBulletin® Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
LinkBacks Enabled by vBSEO © 2011, Crawlability, Inc.
Delphi-PRAXiS (c) 2002 - 2023 by Daniel R. Wolf, 2024 by Thomas Breitkreuz