IdentifiantMot de passe
Loading...
Mot de passe oublié ?Je m'inscris ! (gratuit)
Navigation

Inscrivez-vous gratuitement
pour pouvoir participer, suivre les réponses en temps réel, voter pour les messages, poser vos propres questions et recevoir la newsletter

Contribuez MySQL Discussion :

Obtenir les n premiers éléments de chaque catégorie


Sujet :

Contribuez MySQL

  1. #1
    ced
    ced est déconnecté
    Rédacteur/Modérateur

    Avatar de ced
    Homme Profil pro
    Gestion de bases de données techniques
    Inscrit en
    Avril 2002
    Messages
    6 016
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 48
    Localisation : France, Loiret (Centre)

    Informations professionnelles :
    Activité : Gestion de bases de données techniques
    Secteur : Agroalimentaire - Agriculture

    Informations forums :
    Inscription : Avril 2002
    Messages : 6 016
    Points : 23 705
    Points
    23 705
    Par défaut Obtenir les n premiers éléments de chaque catégorie
    Bonjour,

    Je vous propose un nouvel élément à utiliser : Obtenir les n premiers éléments de chaque catégorie

    Supposons qu'une table ELEMENT et une table CATEGORIE sont composées comme suit :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    CREATE TABLE `ma_base`.`CATEGORIE` (
    	`id_categorie` INTEGER UNSIGNED NOT NULL AUTO_INCREMENT,
    	`nom_categorie` VARCHAR(45) NOT NULL,
    	PRIMARY KEY (`id_categorie`)
    )
    ENGINE = InnoDB;
     
    CREATE TABLE `ma_base`.`ELEMENT` (
    	`id_element` INTEGER UNSIGNED NOT NULL AUTO_INCREMENT,
    	`nom_element` VARCHAR(45) NOT NULL,
    	`id_categorie` INTEGER UNSIGNED NOT NULL,
    	PRIMARY KEY (`id_element`),
    	CONSTRAINT `FK_ELT_CAT` FOREIGN KEY `FK_ELT_CAT` (`id_categorie`)
    		REFERENCES `categorie` (`id_categorie`)
    		ON DELETE CASCADE
    		ON UPDATE CASCADE
    )
    ENGINE = InnoDB;
    Le but de la requête est de ramener les n premiers éléments de chaque catégorie par ordre d'identifiant. Pour cela, on fait ce qu'on appelle une division relationnelle, comme suit :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    SELECT
    	e.id_element, e.nom_element, c.id_categorie, c.nom_categorie
    FROM
    	ELEMENT e
    INNER JOIN
    	CATEGORIE c ON e.id_categorie = c.id_categorie
    WHERE (
    	SELECT COUNT(*)
        FROM ELEMENT e1
        WHERE e1.id_categorie = e.id_categorie
        AND e1.id_element < e.id_element
    ) < n
    Il suffit de remplacer n par la valeur souhaitée.

    Pour obtenir les n derniers éléments de chaque catégorie, il suffit d'inverser le sens de l'inégalité dans la sous-requête :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    SELECT
    	e.id_element, e.nom_element, c.id_categorie, c.nom_categorie
    FROM
    	ELEMENT e
    INNER JOIN
    	CATEGORIE c ON e.id_categorie = c.id_categorie
    WHERE (
        SELECT COUNT(*)
        FROM ELEMENT e1
        WHERE e1.id_categorie = e.id_categorie
        AND e1.id_element > e.id_element
    ) < n
    Attention, cette requête ne fonctionne qu'avec une version de MySQL supportant les sous-requêtes (version 4.1 ou postérieure).

    Qu'en pensez-vous ?
    Rédacteur / Modérateur SGBD et R
    Mes tutoriels et la FAQ MySQL

    ----------------------------------------------------
    Pensez aux balises code et au tag
    Une réponse vous a plu ? N'hésitez pas à y mettre un
    Je ne réponds pas aux questions techniques par message privé, les forums sont là pour ça

  2. #2
    Membre émérite
    Avatar de gene69
    Profil pro
    Inscrit en
    Janvier 2006
    Messages
    1 769
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations professionnelles :
    Secteur : High Tech - Produits et services télécom et Internet

    Informations forums :
    Inscription : Janvier 2006
    Messages : 1 769
    Points : 2 446
    Points
    2 446
    Par défaut
    ça à l'air sympa. ya l'équivalent d'une base scott/tiger pour mysql? je veux dire la même chose avec des données.

    point de vue calculs, c'est gourmand?
    PHP fait nativement la validation d'adresse électronique .
    Celui qui a inventé mysql_connect(...) or die() est déjà mort plusieurs fois.

    Utilisez le bouton résolu!

  3. #3
    Membre émérite Avatar de pacmann
    Homme Profil pro
    Consulté Oracle
    Inscrit en
    Juin 2004
    Messages
    1 626
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Bas Rhin (Alsace)

    Informations professionnelles :
    Activité : Consulté Oracle
    Secteur : Distribution

    Informations forums :
    Inscription : Juin 2004
    Messages : 1 626
    Points : 2 845
    Points
    2 845
    Par défaut
    Salut !

    Je trouve ça juste un peu dommage que l'exemple donne un tri sur une PK technique

    Genre j'aurais bien vu par exemple une date de création à la place, et combiner l'ordre date_creation + increment (pour départager les ex-aequos et garantir qu'on en sorte au plus n)

    (c'est ma photo)
    Paku, Paku !
    Pour les jeunes incultes : non, je ne suis pas un pokémon...

    Le pacblog : http://pacmann.over-blog.com/

  4. #4
    Modérateur

    Avatar de CinePhil
    Homme Profil pro
    Ingénieur d'études en informatique
    Inscrit en
    Août 2006
    Messages
    16 799
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 60
    Localisation : France, Haute Garonne (Midi Pyrénées)

    Informations professionnelles :
    Activité : Ingénieur d'études en informatique
    Secteur : Enseignement

    Informations forums :
    Inscription : Août 2006
    Messages : 16 799
    Points : 34 031
    Points
    34 031
    Billets dans le blog
    14
    Par défaut
    Je me suis posé la même question tout récemment car j'ai eu à sortir un top 5 selon deux critères, l'un classé en ASC et l'autre en DESC.

    Comme il n'y a pas beaucoup de catégories, je suis passé par des requêtes UNION et LIMIT 5. Un truc du genre :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    SELECT tmp.les_colonnes
    FROM
    (
    	(
    		SELECT la_categorie, les_colonnes
    		FROM la_table
    		WHERE la_categorie = 1
    		ORDER BY un_critere ASC, un_autre_critere DESC
    		LIMIT 5
    	)
    	UNION ALL
    	(
    		SELECT la_categorie, les_colonnes
    		FROM la_table
    		WHERE la_categorie = 2
    		ORDER BY un_critere ASC, un_autre_critere DESC
    		LIMIT 5
    	)
    	UNION ALL
    -- etc
    ) tmp
    ORDER BY la_categorie, un_critere ASC, un_autre_critere DESC
    Philippe Leménager. Ingénieur d'étude à l'École Nationale Supérieure de Formation de l'Enseignement Agricole. Autoentrepreneur.
    Mon ancien blog sur la conception des BDD, le langage SQL, le PHP... et mon nouveau blog sur les mêmes sujets.
    « Ce que l'on conçoit bien s'énonce clairement, et les mots pour le dire arrivent aisément ». (Nicolas Boileau)
    À la maison comme au bureau, j'utilise la suite Linux Mageïa !

  5. #5
    Membre émérite Avatar de pacmann
    Homme Profil pro
    Consulté Oracle
    Inscrit en
    Juin 2004
    Messages
    1 626
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 43
    Localisation : France, Bas Rhin (Alsace)

    Informations professionnelles :
    Activité : Consulté Oracle
    Secteur : Distribution

    Informations forums :
    Inscription : Juin 2004
    Messages : 1 626
    Points : 2 845
    Points
    2 845
    Par défaut
    Quand tu as deux (ou plus) critères, il faut réaliser les combinaisons en priorisant.

    Dans ton cas, la condition du count(*) de ced serait un truc du genre :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    WHERE a.un_critere < b.un_critèreASC OR (a.un_critere = b.un_critere AND a.un_autre_critere > b.un_autre_critere)

    (c'est ma photo)
    Paku, Paku !
    Pour les jeunes incultes : non, je ne suis pas un pokémon...

    Le pacblog : http://pacmann.over-blog.com/

  6. #6
    Membre éprouvé

    Profil pro
    Inscrit en
    Juillet 2006
    Messages
    1 448
    Détails du profil
    Informations personnelles :
    Localisation : Belgique

    Informations forums :
    Inscription : Juillet 2006
    Messages : 1 448
    Points : 1 234
    Points
    1 234
    Par défaut
    Ça ressemble (en même temps il n'y a pas des milliers de possibilité en MySQL) à un query que j'ai écrit récemment.
    Il y a une subtilité supplémentaire dans mon query, c'est une optimisation prenant tout son sens lorsque des critères de recherche impliquent des jointures.

    En effet, si des lignes doivent être éliminées du résultats, elles doivent l'être autant dans le sous query qui calcul l'ordre de la ligne (dépendante d'un tri) que dans le query principal.
    Pour ne pas répéter deux fois des jointures qui ne serviraient qu'au filtrage, je procède donc ainsi :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
     
    SELECT
    	H.id
    	, H.stockID
    	, H.value
    	, H.date
    	, H.userID
    	, U.name AS userName
    FROM #__mystock_history AS H
    LEFT JOIN #__users AS U ON (U.id = H.userID)
    WHERE H.stockID IN (%1$s)
    AND (
    	SELECT
    		 COUNT(*) * SUM(CASE WHEN H.id = H2.id THEN 1 ELSE 0 END)
    	FROM #__mystock_history AS H2
    	LEFT JOIN #__mystock_cancel AS C1 ON (C1.canceledID = H2.id)
    	LEFT JOIN #__mystock_cancel AS C2 ON (C2.correctionID = H2.id)
    	WHERE H2.stockID = H.stockID
    	AND C1.canceledID IS NULL
    	AND C2.correctionID IS NULL
    	AND H2.Date >= H.Date
    	AND (
    		H2.Date > H.Date
    		OR
    		H2.id >= H.id
    	)
    ) BETWEEN 1 AND %2$d
    ORDER BY H.stockID ASC, H.date DESC
    %1$s = une liste d'identifiant séparés par des virgules (ex "1,2,3,4")
    %2$d = Le nombre maximale de lignes à obtenir par "groupe"

    L'idée c'est d'obtenir dans le sous query une formule qui puisse m'inidquer deux choses :
    1) l'ordre de la ligne
    2) si la ligne convient au filtre

    Le sous query renverra 0 si la ligne ne convient pas au filtre et l'ordre de la ligne sinon.
    Most Valued Pas mvp

  7. #7
    Rédacteur

    Avatar de SQLpro
    Homme Profil pro
    Expert bases de données / SQL / MS SQL Server / Postgresql
    Inscrit en
    Mai 2002
    Messages
    21 763
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Var (Provence Alpes Côte d'Azur)

    Informations professionnelles :
    Activité : Expert bases de données / SQL / MS SQL Server / Postgresql
    Secteur : Conseil

    Informations forums :
    Inscription : Mai 2002
    Messages : 21 763
    Points : 52 554
    Points
    52 554
    Billets dans le blog
    5
    Par défaut
    Elle n'est valable QUE dans le cas ou vous utilisez une clef primaire comme comparaison d’inégalité. Si vous utilisez une colonne quelconque, cela peut conduire à des résultats faux.

    Dans ce ac, seule la fonction ROW_NUMBER() ou RANK(), implémentée dans tous les SGBDR sauf MySQL sera la seule solution.

    A +
    Frédéric Brouard - SQLpro - ARCHITECTE DE DONNÉES - expert SGBDR et langage SQL
    Le site sur les SGBD relationnels et le langage SQL: http://sqlpro.developpez.com/
    Blog SQL, SQL Server, SGBDR : http://blog.developpez.com/sqlpro
    Expert Microsoft SQL Server - M.V.P. (Most valuable Professional) MS Corp.
    Entreprise SQL SPOT : modélisation, conseils, audit, optimisation, formation...
    * * * * * Expertise SQL Server : http://mssqlserver.fr/ * * * * *

  8. #8
    Membre éprouvé

    Profil pro
    Inscrit en
    Juillet 2006
    Messages
    1 448
    Détails du profil
    Informations personnelles :
    Localisation : Belgique

    Informations forums :
    Inscription : Juillet 2006
    Messages : 1 448
    Points : 1 234
    Points
    1 234
    Par défaut
    En MySQL, vous pouvez définir une variable entière à 0 et renvoyer sa valeur tout en l'incrémentant dans un select ordonné (order by...).
    Je ne me souviens pas de la syntaxe exacte, mais c'est une chose à connaître dans ces cas-là.

    Note : Quel déterrement !
    Most Valued Pas mvp

  9. #9
    Membre expert
    Avatar de alassanediakite
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Août 2006
    Messages
    1 599
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 46
    Localisation : Mali

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Août 2006
    Messages : 1 599
    Points : 3 590
    Points
    3 590
    Billets dans le blog
    8
    Par défaut
    Salut
    Un exemple avec variable initialisée à zéro
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    SET @rank=0;
    SELECT @rank:=@rank+1 as rank, ID, Name FROM city;
    @+
    Le monde est trop bien programmé pour être l’œuvre du hasard…
    Mon produit pour la gestion d'école: www.logicoles.com

Discussions similaires

  1. Réponses: 1
    Dernier message: 04/09/2010, 12h07
  2. [AC-2007] Renvoyer les n éléments de chaque catégorie
    Par triaguae dans le forum Requêtes et SQL.
    Réponses: 17
    Dernier message: 19/04/2010, 14h28
  3. Renvoyer les n éléments de chaque catégorie
    Par triaguae dans le forum Requêtes
    Réponses: 7
    Dernier message: 10/04/2010, 22h07
  4. Réponses: 3
    Dernier message: 07/03/2010, 20h50
  5. Réponses: 2
    Dernier message: 07/03/2007, 16h30

Partager

Partager
  • Envoyer la discussion sur Viadeo
  • Envoyer la discussion sur Twitter
  • Envoyer la discussion sur Google
  • Envoyer la discussion sur Facebook
  • Envoyer la discussion sur Digg
  • Envoyer la discussion sur Delicious
  • Envoyer la discussion sur MySpace
  • Envoyer la discussion sur Yahoo