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

Requêtes MySQL Discussion :

Listing à grouper par mois


Sujet :

Requêtes MySQL

  1. #1
    Futur Membre du Club
    Profil pro
    Inscrit en
    Décembre 2004
    Messages
    22
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2004
    Messages : 22
    Points : 7
    Points
    7
    Par défaut Listing à grouper par mois
    Bonjour à tous,

    Voilà mon soucis.

    J'ai une table un listing d'interventions (et leurs durées en minutes) faites chez des clients. Ce qui donne à peu près ça :

    ID | Client | Duree | Date
    1 | SARL_duchemin | 55 | 2014-10-29
    2 | SARL_duchemin | 35 | 2014-09-24
    3 | SA_martin | 65 | 2014-09-22
    4 | SA_martin | 15 | 2014-10-12
    5 | SARL_duchemin | 12 | 2014-10-26
    6 | SA_martin | 75 | 2014-10-20

    ...

    J'ai besoin de sortir un tableau par année, sur lequel je puisse voir pour chaque mois de l'année (donc même celles qui ne comportent aucune intervention), combien j'ai eu d'interventions (et la durée totale).
    Quelque chose comme :

    Client | Janvier | Février | Mars | Avril | Mai | Juin | Juillet | Août | Septembre | Octobre | Novembre | Décembre | TOTAL
    SARL_duchemin | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 2 (1h07) | 1 (35mn) | 0 | 0 | 3 (1h42)
    SA_martin | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 2 (1h30) | 1 (1h05) | 0 | 0 | 3 (2h35)


    Et je suis en train de perdre la raison... Je fais des jointures dans tous les sens, je lance des requêtes dans les requêtes, des for, des if... Alors qu'il doit y avoir une simple requête SQL pour arriver à me sortir ça... ?
    ...Non ?

    D'avance merci pour toute aide !

  2. #2
    Membre averti
    Homme Profil pro
    Inscrit en
    Juillet 2012
    Messages
    200
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : Bénin

    Informations forums :
    Inscription : Juillet 2012
    Messages : 200
    Points : 342
    Points
    342
    Par défaut
    Bonsoir.
    Je pense que vous avez trois options :
    - Utiliser le partitionnement de table. Un excellent article d'introduction ici. Les infos pour MySQL ici.
    - La seconde option est d'utiliser une procédure stockée qui se charge de ce travail et renvoie le jeu de résultats.
    - Une autre solution serait d'avoir une table "mois" avec les attributs "date_debut" et "date_fin" et qui contient les 12 mois avec leur date de début et de fin. Et faire une requête qui sélectionne les interventions situées dans les intervalles de chaque mois.

  3. #3
    Expert confirmé
    Homme Profil pro
    Inscrit en
    Mai 2002
    Messages
    3 173
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 41
    Localisation : France, Rhône (Rhône Alpes)

    Informations forums :
    Inscription : Mai 2002
    Messages : 3 173
    Points : 5 345
    Points
    5 345
    Par défaut
    bonjour,


    une requete type pour pivoter :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
     
    select col_groupement,
    sum(case when ma_cond between X and Y then 1 else 0 end),
    sum(case when ma_cond between W and Z then 1 else 0 end),
    ....
    from ma_table
    group by col_groupement

    Sinon votre table est mal modélisée, on ne met pas le nom du client en dur mais on fait référence à une table client.

    Vous aurez un gain de performance si votre table est grosse en faisant comme ceci.

  4. #4
    Membre émérite Avatar de Drizzt [Drone38]
    Homme Profil pro
    Directeur de projet
    Inscrit en
    Mai 2004
    Messages
    1 001
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 41
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Directeur de projet

    Informations forums :
    Inscription : Mai 2004
    Messages : 1 001
    Points : 2 453
    Points
    2 453
    Par défaut
    Bonjour,

    Ce que tu souhaites faire c'est de la cosmétique, ton SGBD n'est pas là pour cela.
    Tu peux très facilement obtenir un résultat de la forme suivante :

    SARL_Duchemin Septembre 1 35
    SARL_Duchemin Octobre 2 65
    SARL_Martin Septembre 1 65
    ....
    En effectuant un GROUP BY sur la date tronquée au premier du mois.
    Tu peux éventuellement ajouter facilement les lignes des mois à 0 en effectuant une jointure sur une table calendrier.

    Le passage en colonne, c'est de la cosmétique et c'est à ton programme appelant de s'en charger. Idem pour le total.

    C'est bien de vouloir tout faire faire par le SGBD mais il faut se limiter à ce qu'il sait bien faire.
    Si vraiment tu n'as pas le choix, la solution de punkoff répondera à ton besoin.
    Je ne réponds pas aux questions techniques par MP, le forum est là pour cela.

    La crypto c'est comme les flambys, une fois que tu as trouvé la languette tu as juste à tirer pour tout faire tomber.

    (\ _ /)
    (='.'=)
    Voici Lapinou. Aidez le à conquérir le monde
    (")-(") en le reproduisant

  5. #5
    Membre éprouvé
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Août 2012
    Messages
    631
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Août 2012
    Messages : 631
    Points : 1 220
    Points
    1 220
    Par défaut
    en reprenant tout ce qui a été dit précédemment,pour trouver le nombre total d'intervention pour chaque mois(y compris le mois sans intervention) de l'année tu peux identifier le mois(avec MONTH(Date) ) puis faire le comptage de toutes les interventions de ce mois.Et si aucune intervention n'a lieu pour ce mois on renvoie 0 (avec COUNT(NULL)).

    Par exemple pour compter le nombre d'intervention du mois de janvier pour chaque année:

    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    SELECT Client YEAR(Date) as an , COUNT(CASE WHEN MONTH(Date) = 1 THEN 1 ELSE NULL END) jan , COUNT(*) AS nb_intervention_par_an GROUP BY an;

    Pour obtenir la durée totale des interventions pour le mois de janv
    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    SELECT Client YEAR(Date) as an , SUM(CASE WHEN MONTH(Date) = 1 THEN Duree ELSE 0 END) jan , COUNT(*) AS nb_intervention_par_an GROUP BY an;

    On peut cumuler les 2 requêtes précédentes pour obtenir la durée et le nombre d'interventions pour les mois de janv, février en concatenant la durée et le nombre d'interventions:
    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
     
    SELECT Client YEAR(Date) as an ,
     CONCAT(SUM(CASE WHEN MONTH(Date) = 1 THEN Duree ELSE 0 END),'|',COUNT(CASE WHEN MONTH(Date) = 1 THEN 1 ELSE NULL END)) jan, 
     CONCAT(SUM(CASE WHEN MONTH(Date) = 2 THEN Duree ELSE 0 END),'|',COUNT(CASE WHEN MONTH(Date) = 2 THEN 1 ELSE NULL END)) fev, 
    //...jusqu'à décembre
     COUNT(*) AS nb_intervention_par_an GROUP BY an;
    Pour ne pas avoir à répéter les 12 mois, on peut construire un tableau des mois qui permettra de construire dynamiquement la requête
    Code php : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
     
    $tabMois=array(1=>'jan',2=>'fev',3=>'mar', 4=>'avr',5=>'mai',6=>'juin',7=>'juill',8=>'au'); //à compléter jusqu'au 12 ème mois
     $somme_duree_et_total_intervention='';
     
    foreach( $tabMois as $k=>$mois ){
     $somme_duree_et_total_intervention.=", CONCAT(SUM(CASE WHEN MONTH(Date) = $k THEN Duree ELSE 0 END),'|',COUNT(CASE WHEN MONTH(Date) = $k THEN 1 ELSE NULL END)) $mois";            
    }
     
    $sql="SELECT Client YEAR(Date) as an $somme_duree_et_total_intervention, COUNT(*) AS nb_intervention_par_an from Client GROUP BY an ";
    //pour afficher la requête
    echo $sql;

  6. #6
    Futur Membre du Club
    Profil pro
    Inscrit en
    Décembre 2004
    Messages
    22
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2004
    Messages : 22
    Points : 7
    Points
    7
    Par défaut
    Bonjour tout le monde !

    Merci pour ces nombreuses réponses toutes utiles !

    Alors pour info oui j'ai mes clients dans une autre table, mais c'était pour vous simplifier l'exemple.

    Je suis parti du résumé de armel18, mais la requête ne passe pas, je n'arrive pas à voir ce qui cloche...

    Voilà ce que j'ai comme requête, en simplifiant sur les derniers mois de l'année uniquement :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    SELECT id_client, CONCAT(SUM(CASE WHEN MONTH(date) = 9 THEN duree ELSE 0 END),'|',COUNT(CASE WHEN MONTH(date) = 9 THEN 1 ELSE NULL END)) as sep, CONCAT(SUM(CASE WHEN MONTH(date) = 10 THEN duree ELSE 0 END),'|',COUNT(CASE WHEN MONTH(date) = 10 THEN 1 ELSE NULL END)) as oct, CONCAT(SUM(CASE WHEN MONTH(date) = 11 THEN duree ELSE 0 END),'|',COUNT(CASE WHEN MONTH(date) = 11 THEN 1 ELSE NULL END)) as nov, CONCAT(SUM(CASE WHEN MONTH(date) = 12 THEN duree ELSE 0 END),'|',COUNT(CASE WHEN MONTH(date) = 12 THEN 1 ELSE NULL END)) as dec, COUNT(*) AS nb_intervention_par_an FROM ri_rapports WHERE YEAR(date)= 2014
    On est bien d'accord qu'il fallait ajouter un "as" devant le nom du mois hein ?
    Et j'ai un select pour envoyer l'année sur laquelle on veut faire la recherche, donc j'ai ajouté un WHERE.

  7. #7
    Membre éprouvé
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Août 2012
    Messages
    631
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Août 2012
    Messages : 631
    Points : 1 220
    Points
    1 220
    Par défaut
    aucune erreur affichée?
    ce qui sûr le mot dec est un mot réservé.
    je revois ta requête en fin de journée

  8. #8
    Futur Membre du Club
    Profil pro
    Inscrit en
    Décembre 2004
    Messages
    22
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2004
    Messages : 22
    Points : 7
    Points
    7
    Par défaut
    Bah cherche pas plus loin c'était bien ça !
    La requête passe, j'avance !

  9. #9
    Futur Membre du Club
    Profil pro
    Inscrit en
    Décembre 2004
    Messages
    22
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2004
    Messages : 22
    Points : 7
    Points
    7
    Par défaut
    Donc là ou ça bloque ensuite, c'est qu'il m'affiche toutes les interventions cumulées pour tous les clients, et pas par client (une seule ligne donc).

  10. #10
    Expert confirmé
    Homme Profil pro
    Inscrit en
    Mai 2002
    Messages
    3 173
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Âge : 41
    Localisation : France, Rhône (Rhône Alpes)

    Informations forums :
    Inscription : Mai 2002
    Messages : 3 173
    Points : 5 345
    Points
    5 345

  11. #11
    Futur Membre du Club
    Profil pro
    Inscrit en
    Décembre 2004
    Messages
    22
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Décembre 2004
    Messages : 22
    Points : 7
    Points
    7
    Par défaut
    GROUP BY id_client...

    Bien sûr...

    Bon et bien merci à tous, a priori je peux avancer ! :-)

  12. #12
    Membre éprouvé
    Homme Profil pro
    Développeur informatique
    Inscrit en
    Août 2012
    Messages
    631
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France

    Informations professionnelles :
    Activité : Développeur informatique

    Informations forums :
    Inscription : Août 2012
    Messages : 631
    Points : 1 220
    Points
    1 220
    Par défaut
    Citation Envoyé par bibos Voir le message
    GROUP BY id_client...

    Bien sûr...

    Bon et bien merci à tous, a priori je peux avancer ! :-)
    suis désolé j'avais oublié de mettre l'id_client dans le GROUP BY.
    le principe pour une requête avec groupement est que tous les champs(exceptés les fonctions d'agrégation comme SUM(),AVG()...) présents dans le SELECT doivent l'être aussi dans le GROUP BY.

+ Répondre à la discussion
Cette discussion est résolue.

Discussions similaires

  1. Grouper par mois à l'aide de EXTRACT
    Par QGRdev dans le forum Langage SQL
    Réponses: 5
    Dernier message: 23/12/2012, 17h44
  2. grouper par mois un timestamp
    Par broule dans le forum Requêtes
    Réponses: 1
    Dernier message: 13/09/2010, 16h23
  3. grouper par mois dans une listview
    Par superkiller dans le forum Windows Presentation Foundation
    Réponses: 3
    Dernier message: 24/01/2010, 14h25
  4. [Interbase] selectionner et grouper par mois ?
    Par makaphrodite dans le forum Langage SQL
    Réponses: 8
    Dernier message: 04/11/2008, 15h20
  5. [MySQL] Grouper par mois
    Par Friedrick dans le forum PHP & Base de données
    Réponses: 3
    Dernier message: 17/09/2008, 15h57

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