Hallo,

Ik heb een vraag over Mysql. Ik wil op basis van uitslagen een stand opmaken, maar ik kom er niet uit. De puntentelling is enigszins gecompliceerd, en ik weet niet of ik deze het beste in een table kan stoppen of in de query zelf.

Even een korte uiteg. De stand wordt gebaseerd op de punten behaald in het afgelopen jaar, een beetje zoals in het tennis ook gebeurd. Er zijn twee categoriën A en B, en vier niveau's wedstrijden (N1,N2,N3,N4). Daarnaast is er nog een extra moeilijkheid, namelijk dat van de wedstrijden van niveau N3 en N4 alleen de beste 5 resultaten uit het afgelopen jaar tellen.

De code zou er - voor zover mijn kennis reikt - dus ongeveer zo moeten uitzien denk ik:

SELECT * FROM speler AS s
JOIN uitslagen AS u ON s.id = u.uitslag_naam
JOIN kalender AS k ON k.categorie = u.uitslag_categorie AND k.seizoen = u.uitslag_seizoen AND k.kalender_id = u.uitslag_id

....

GROUP BY uitslag_id
AND datum BETWEEN '".$datum."' - INTERVAL 1 YEAR AND '".$datum."'
ORDER BY datum DESC


'Datum' komt uit kalender, en $datum haal ik op uit de URL met de GET-functie. Omdat elke wedstrijd maar één keer meetelt moet Group BY uitslag_id ervoor zorgen dat er van elke wedstrijd maar één resultaat is, en door middel van de Order by datum de recentste editie. Al moet de stand uiteindelijk worden 'georderd' op basis van degene met de meeste punten. Ik heb al veel gezocht ivm het rekenwerk, maar ik weet niet hoe dat moet, dus dat heb ik maar even open gelaten.

De puntentelling is als volgt:

uitslag_categorie / uitslag_niveau / puntenverdeling
A - N1 / 20 - 18 - 16 - 14 - 12 - 10 - 8 - 6 - 4 - 2
A - N2 / 10 - 9 - 8 - 7 - 6 - 5 - 4 - 3 - 2 - 1
A - N3 / 5 - 4 - 3 - 2 - 1
B - N4 / 3 - 2 - 1

N1 - 1 van 1
N2 - 3 van 3
N3 - 5 van X (alleen vijf beste resultaten tellen mee)
N4 - 5 van X (alleen vijf beste resultaten tellen mee)


Even uit de vrije hand dus zoiets:
IF uitslag_categorie = A AND uitslag_niveau = N1 AND uitslag_positie = 1 --> 20

Ik heb ook een table 'puntentelling' gemaakt bestaande uit de kolommen categorie - niveau - p1 - p2 - p3 etc. maar ik weet niet of dat de manier is waarop dat zou moeten.


Ik hoop dat jullie mij kunnen helpen!
Gezien de complexiteit zou ik zelf een apart script maken om de stand te berekenen en op te slaan in een standen tabel.
Het script maakt dan eerste de standentabel leeg (waarschijnlijk voor het betreffende seizoen) en loopt vervolgens de andere tabellen af om per speler het aantal punten te berekenen. Dit sla je vervolgens op.
Je kunt dan de standen tabel laten zien gesorteerd op punten van hoog naar laag.

Hoe de berekening precies moet kan ik uit je vraagstelling niet helemaal opmaken, maar misschien geeft je dit een idee hoe verder te gaan.

De standentabel zou iets kunnen zijn als:
- id
- seizoen_id
- speler_id
- punten


Zelf een apart script ja.

En ik zou zelf niet alles in 1 query gooien.

Stuk code om de beste 5 van N3 en N4 uit de database halen.
Stuk code voor N1.
Stuk code voor N2.
Stuk code tussendoor voor A of B? (begrijp de structuur ook niet helemaal).
En dan een stuk code om de uitslagen van die codes op te tellen.

Lijkt me dat het wel te doen is als je de berekening gewoon in hapklare brokjes opbreekt.
Bedankt voor de reacties. Ik ben pas recentelijk begonnen met php en mysql, dus het is allemaal vrij nieuw voor me, dus ik weet nog niet goed wat de mogelijkheden zijn en wat jullie precies bedoelen.

Als ik het goed begrijp heb ik dus een PHP-script nodig die gegevens uit een table haalt, en daaruit vervolgens een nieuwe table aanmaakt? Ik heb tot op heden alleen maar tables gebruikt waarin ik zelf de inhoud heb aangegeven, zijn er voorbeelden van hoe zo'n script eruit zou moeten zien om een 'gegenereerde' table te maken?

Is het anders ook mogelijk om een extra kolom in de table 'uitslagen' te genereren voor de punten? In principe staan daar namelijk al alle gegevens die ik nodig denk te hebben, alleen wil ik de punten niet bij elke uitslag handmatig invoeren. Dus iets als: IF uitslag_categorie = A AND uitslag_niveau = N1 AND uitslag_positie = 1 --> kolom punten --> 20?

Over de puntentelling:
A is senioren
B is junioren
Daarbinnen zijn dus diverse niveau's wedstrijden (N1,N2,N3,N4) met een eigen puntentelling.
In uitslagen staat dus welk ID (id/uitslag_naam) in welk uitslag_seizoen, uitslag_categorie, en uitslag_niveau welke uitslag_postie behaalde.

@Jan de Laet: Ik wil alleen dat je per datum kan zien wat de stand op die datum was. Dat kan denk ik niet als je een 'totaal' gebruikt.

@L deB: Hoe moet ik dit voor me zien? Bedoel je met SELECT in SELECT (heb ik wel eens voorbij zien komen)?
Misschien is het wel voldoende om de punten als extra veld bij de uitslagen te zetten. Ik weet alleen niet hoe je dan voor elkaar krijgt omdat voor N3 en N4 maar voor 5 wedstrijden te doen.
En als ik je goed begrijp wil je daarna het totaal van de punten laten zien van de spelers op 1 datum, dus dan heb je een SELECT speler_id, SUM(punten) FROM uitslagen WHERE datum = 20145-12-02' GROUP BY speler_id nodig.

Een eventuele standentabel kun je in het script maken met CREATE TABLE, maar waarschijnlijk is het handiger om de tabel buiten het script te maken en als je de stand berekent voor een datum de oude informatie te verwijderen, bijv DELETE FROM standen WHERE datum='2015-12-02'.

Ik ken nog niet alle mogelijkheden met Mysql, maar kan je niet Selecten in een Select? Ik dacht dat ik die constructie heb zien passeren. Je zou dan denk ik voor N3 en N4 aflopend kunnen sorteren op punten en dan LIMIT 5 doen? Dan krijg je de beste 5 resultaten toch.

Dat is inderdaad precies wat ik bedoel.

Maar hoe zou ik die punten automatisch als extra veld bij uitslagen kunnen krijgen?
Tabellen maken en velden toevoegen zijn meestal eenmalig, die doe je het beste met een tool als phpMyAdmin (bij mijn server is dat vanuit cPanel, de beheertool, beschikbaar). Je hebt met phpMyAdmin een vrij eenvoudige tool om met hulp de CREATE TABEL en ALTER TABLE uit te voeren, bijv ALTER TABLE `uitslagen` ADD `punten` INT NOT NULL DEFAULT '0';


Select in Select: in principe kun je daar waar een 'FROM table' staat, ipv de table ook een '(SELECT a, b, c FROM table2 WHERE a=1)'. Men noemt dat subqueries. Het resultaat tussen de haakjes gedraagt zich dan alsof het een bestaande table was.
Maar dat heeft meestal alleen zin als je met het tussenresultaat nog wat meer wil doen. Bijv.


SELECT * FROM (SELECT a, b, SUM(c) FROM table2 WHERE a=1 GROUP BY a, b ORDER BY SUM(c) DESC) AS t
JOIN spelers ON spelers.id = t.a

Hier koppel je dus het resultaat t van de subquery aan de spelers tabel.


N3 en N4
Omdat jij van N1 en N2 alles wilt hebben en van N3 en N4 alleen de laatste 5, kun jo overwegen met UNION te werken. Daarmee 'verenig' je meerdere SELECTs tot 1 resultaat.


SELECT * FROM uitslagen 
    WHERE uitslag_niveau in ("N1", "N2")   -- alles van N1 en N2
UNION
SELECT * FROM (SELECT * FROM uitslagen 
               WHERE uitslag_niveau = "N3" 
               AND datum BETWEEN '".$datum."' - INTERVAL 1 YEAR AND '".$datum."
               ORDER BY datum DESC LIMIT 5) AS u2            -- laatste 5 van N3
UNION
SELECT * FROM (SELECT * FROM uitslagen 
               WHERE uitslag_niveau = "N4" 
               AND datum BETWEEN '".$datum."' - INTERVAL 1 YEAR AND '".$datum."
               ORDER BY datum DESC LIMIT 5) AS u3            -- laatste 5 van N4


Het zal niet helemaal correct zijn que syntax, maar misschien brengt het je op een idee.
Jan de Laet op 03/12/2015 08:24:22

Tabellen maken en velden toevoegen zijn meestal eenmalig, die doe je het beste met een tool als phpMyAdmin (bij mijn server is dat vanuit cPanel, de beheertool, beschikbaar). Je hebt met phpMyAdmin een vrij eenvoudige tool om met hulp de CREATE TABEL en ALTER TABLE uit te voeren, bijv ALTER TABLE `uitslagen` ADD `punten` INT NOT NULL DEFAULT '0';


Ik heb ook phpMyAdmin en heb de kolom punten toegevoegd. Mijn vraag was eigenlijk meer: Hoe kan ik in een rij de kolom 'punten' automatisch gevuld krijgen? In elke rij staan de categorie, het niveau, en de uitslag, maar kan ik met die gegevens het juiste puntentotaal genereren in 'punten'?

UPDATE uitslagen
    SET punten = 
        CASE WHEN (uitslag_categorie = "A" AND uitslag_niveau = "N1" AND uitslag_positie = 1) THEN 20
             WHEN (uitslag_categorie = "A" AND uitslag_niveau = "N1" AND uitslag_positie = 2) THEN 18
             enz
             ELSE 0
        END
WHERE ...
Ik weet niet of ik je helemaal goed begrijp, maar ik heb het idee dat je wil dat op elk moment het puntentotaal klopt en dit automatisch gedaan wordt.. dit wil niet automatisch, je moet er wel een code voor schrijven en deze laten uitvoeren.

Dus zoals Jan aangeeft voor N3 en N4 en op die manier alle juiste berekeningen uitvoeren. Vervolgens per resultaat (of het geheel van die resultaten) in de rij 'punten' plaatsen met een query met INSERT (edit: UPDATE inderdaad).

En even terugkomende op wat ik eerder zei en je naar vroeg: ik bedoelde niet een insert binnen een insert. Ik bedoelde gewoon dat het blijkbaar lastig voor je is om in 1 code alle berekeningen uit te laten voeren dus dan lijkt het me beter om de berekeningen in kleine stukjes op te delen. één die voor N4 berekent. één die voor N3 berekent enzovoort. Heb je een wat langere code dat klopt maar is wel gemakkelijker (althans voor mij).
Ik heb de UPDATE in een query gezet en in een .php pagina en het werkt! Supervet! Als ik het goed begrijp moet ik (of iemand anders) dus die pagina bekijken en dan wordt het geupdate?

Ik ben inmiddels alweer wat opgeschoten, en het optellen lukt heel aardig, maar ik loop tegen enkele probleempjes aan.

Ik heb nu dit:
$sql = "SELECT uitslag_naam, SUM(punten) AS punt FROM uitslagen AS u
 JOIN kalender AS k ON u.uitslag_categorie = k.categorie AND u.uitslag_seizoen = k.seizoen AND u.uitslag_id = k.kalender_id 
WHERE datum BETWEEN '".$datum."' - INTERVAL 1 YEAR AND '".$datum."' 
AND uitslag_niveau IN ('N1', 'N2') AND punten > 0
GROUP BY uitslag_naam
ORDER BY punt DESC"


- Het kan voorkomen dat een wedstrijd het ene jaar in december is, en het andere jaar in november. Ik wil dus eigenlijk zorgen dat alleen de laatste editie (de recentste datum) van de uitslag_id erin komt. Ik doe alleen al Group By uitslag_naam (zodat de ID wordt gegroepeerd), en ook nog Group By uitslag_id werkt niet.

- Hoe krijg ik die Union er goed in voor N3 en N4, en hoe moet ik dat vervolgens optellen?

Reageren