Maison > Questions et réponses > le corps du texte
Edit 3 - La version MySQL est 8.0.33.
Edit 2 - Voir le code de travail final en bas. Merci @Akina !
J'ai une feuille de pointage pour un événement sportif. Le tableau comporte trois champs associés que je souhaite sélectionner -
scoreID
comme valeur de clé primaireclassifierID
Mappe vers une clé primaire d'une autre table contenant des détails sur une disposition de cours spécifique calculatedPercent
est le résultat d'un événement spécifiqueLa table contient trois autres champs que j'utilise dans la clause WHERE, mais ceux-ci sont accessoires.
Je dois générer une requête qui sélectionne les quatre meilleures valeurs pour calculatedPercent
选择四个最佳值,并规定 classifierID
不能重复。我需要能够捕获 scoreID
et précise que
pour l'utiliser dans les étapes ultérieures du processus.
Voici ma première requête : calculatedPercent
值的行选择了 scoreID
值。然后我注意到有几个成员在同一门课程上获得了第一和第二高分,这违反了 classifierID
SELECT `masterScores`.`scoreID`, `masterScores`.`classifierID`, `masterScores`.`calculatedPercent` FROM `masterScores` WHERE `masterScores`.`memberID` = 3516 AND `masterScores`.`eventDivision` = "O" AND `masterScores`.`scoreUnusable` != "TRUE" ORDER BY `masterScores`.`calculatedPercent` DESC LIMIT 4Au départ, j'ai trouvé que c'était génial car cela sélectionne la ligne avec la valeur
la plus élevée pour un membre donné. Ensuite, j'ai remarqué que plusieurs membres avaient obtenu les première et deuxième notes les plus élevées dans le même cours, ce qui violait l'exigence
de non-duplication des valeurs.J'ai essayé SELECT DISTINCT mais j'ai finalement réalisé que ce dont j'avais vraiment besoin était GROUP BY, j'ai donc fait quelques recherches et découvert que j'obtenais des erreurs liées à only_full_group_by lors de l'exécution de requêtes dans MySql, mais cela n'a pas complètement résolu mon problème.
Ce que j'ai essayé ensuite :
SELECT `masterScores`.`scoreID`, `masterScores`.`classifierID`, MAX(`masterScores`.`calculatedPercent`) AS bestPercent FROM `masterScores` WHERE `masterScores`.`memberID` = 3516 AND `masterScores`.`eventDivision` = "O" AND `masterScores`.`scoreUnusable` != "TRUE" GROUP BY `masterScores`.`classifierID` ORDER BY bestPercent DESC LIMIT 4Voici le message d'erreur suivant :
#1055 - L'expression n°1 de la clause ORDER BY n'est pas dans la clause GROUP BY et contient la colonne non agrégée ".masterScores.calculatedPercent", qui ne dépend pas fonctionnellement des colonnes de la clause GROUP BY ; avec sql_mode=only_full_group_by n'est pas compatiblemasterScores
.scoreID
列使用 MIN 和 MAX,但它与预期不符; scoreID
主键值并不总是所选 calculatedPercent
的值。我在某处读到,因为 scoreID
J'ai envisagé d'utiliser MIN et MAX pour la colonne masterScores
.scoreID
, mais cela ne fonctionne pas comme prévu
sélectionnée ; J'ai lu quelque part que puisque
est la clé primaire, je peux résoudre ce problème en utilisant l'agrégation ANY_VALUE. J'ai essayé ceci :SELECT ANY_VALUE(`masterScores`.`scoreID`), `masterScores`.`classifierID`, MAX(`masterScores`.`calculatedPercent`) AS bestPercent FROM `masterScores` WHERE `masterScores`.`memberID` = 3516 AND `masterScores`.`eventDivision` = "O" AND `masterScores`.`scoreUnusable` != "TRUE" GROUP BY `masterScores`.`classifierID` ORDER BY bestPercent DESC LIMIT 4
qui correspond à la valeur bestPercent. classifierID
选择 1 个 calculatedPercent
和 1 个 scoreID
值。如果不按 classifierID
分组,则每个 classifierID
classifierID
所选的 calculatedPercent
Sélectionne seulement 1
, chaque calculatedPercent
pourrait avoir entre 0 et 400 lignes satisfaisant la clause WHERE, donc je pense que GROUP BY serait approprié ici.
Assurez-vous que le calculatedPercent
sélectionné pour chaque groupe
Garantit scoreID
值实际上代表与选定的 calculatedPercent
la même ligne pour chaque ligne sélectionnée (actuellement, c'est le point auquel le pourcentage est calculé et ma requête échoue).
Ce qui suit est un sous-ensemble des données, par exemple :
ID de score | ID du classificateur | Meilleur pourcentage |
---|---|---|
58007 | 42 | 66.60 |
63882 | 42 | 64.69 |
64685 | 54 | 64.31 |
58533 | 32 | 63.20 |
55867 | 42 | 62.28 |
66649 | 7 | 56.79 |
55392 | 12 | 50.28 |
58226 | 1 | 49.52 |
55349 | 7 | 41.10 |
Voici le résultat souhaité lorsque j'exécute la requête :
ID de score | ID du classificateur | Meilleur pourcentage |
---|---|---|
58007 | 42 | 66.60 |
64685 | 54 | 64.31 |
58533 | 32 | 63.20 |
66649 | 7 | 56.79 |
Voici le résultat réel lorsque j'exécute la requête :
ID de score | ID du classificateur | Meilleur pourcentage |
---|---|---|
55867 | 42 | 66.60 |
64685 | 54 | 64.31 |
58533 | 32 | 63.20 |
55349 | 7 | 56.79 |
Comme le montre la figure, les valeurs scoreID
des première et quatrième lignes de la sortie réelle sont incorrectes.
Pour l’instant, j’accepte toutes les suggestions.
Edit 2 - Solution de travail finale
WITH cte AS ( SELECT scoreID, classifierID, calculatedPercent AS bestPercent, ROW_NUMBER() OVER (PARTITION BY classifierID ORDER BY calculatedPercent DESC, scoreID DESC) AS rn FROM masterScores WHERE memberID = 3516 AND eventDivision = "O" AND scoreUnusable != "TRUE" ) SELECT scoreID, classifierID, bestPercent FROM cte WHERE rn = 1 ORDER BY bestPercent DESC LIMIT 4
J'ai pu tester cela sur six cas où le problème s'est produit, et la solution a résolu chaque problème. Merci encore @Akina !
Marquera ce problème comme résolu.
P粉6968918712023-09-11 13:48:48
SELECT t1.scoreID, classifierID, calculatedPercent AS bestPercent FROM masterScores t1 NATURAL JOIN ( SELECT classifierID, MAX(calculatedPercent) AS calculatedPercent FROM masterScores t2 WHERE memberID = 3516 AND eventDivision = "O" AND scoreUnusable != "TRUE" GROUP BY 1 ORDER BY calculatedPercent DESC LIMIT 4 ) t2
Si (classifierID,calculatedPercent)
不唯一,那么每个classifierID
vous pouvez recevoir plusieurs lignes. Dans ce cas, vous avez besoin de
SELECT MAX(t1.scoreID) AS scoreID, classifierID, calculatedPercent AS bestPercent FROM masterScores t1 NATURAL JOIN ( SELECT classifierID, MAX(calculatedPercent) AS calculatedPercent FROM masterScores t2 WHERE memberID = 3516 AND eventDivision = "O" AND scoreUnusable != "TRUE" GROUP BY 1 ORDER BY calculatedPercent DESC LIMIT 4 ) t2 GROUP BY 2, 3PS. Si votre version de MySQL est 8+, vous devez utiliser
ROW_NUMBER()
dans le CTE au lieu d'une sous-requête.
WITH cte AS ( SELECT scoreID, classifierID, calculatedPercent AS bestPercent, ROW_NUMBER() OVER (PARTITION BY classifierID ORDER BY calculatedPercent DESC, scoreID DESC) AS rn FROM masterScores ) SELECT scoreID, classifierID, bestPercent FROM cte WHERE rn = 1