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

PHP & Base de données Discussion :

PDO - requête d'insertion multiple qui ne fonctionne pas


Sujet :

PHP & Base de données

  1. #1
    Membre averti
    Homme Profil pro
    Étudiant
    Inscrit en
    Mai 2021
    Messages
    30
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2021
    Messages : 30
    Par défaut PDO - requête d'insertion multiple qui ne fonctionne pas
    Bonjour,
    Malgré chatGPT et toutes les ressources d'aujourd'hui, je me confronte à un problème que je n'arrive pas à résoudre dans une requête SQL où j'essaie de faire plusieurs insertions d'un coup.

    Ce code fonctionne très bien :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
     
        $sql = "INSERT INTO postcondition (description, id_user) VALUES (:description, :id_user)";
        $stmt = $this->db->prepare($sql);
        $stmt->execute([
          'description' => $postCondition->getDescription(),
          'id_user' => $postCondition->getUser()->getId()
        ]);
     
        $postCondition->setId($this->db->lastInsertId());
        return $postCondition;
    Sauf que des postconditions, j'en ai plusieurs, et jusqu'à présent, j'exécutais donc ce code dans un foreach. Je multipliais donc les requêtes à la base de données. Dans un soucis de performances, j'ai eu envie de faire une seule requête, qui intègre toutes les lignes d'un coup.

    J'ai donc revu mon code, de cette manière :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
     
        $sql = "INSERT INTO postcondition (description, id_user) VALUES ";
        $i = 0;
        $data = [];
        foreach ($postConditions as $postCondition) {
          $sql .= " (:description" . $i . ", :id_user" . $i . "),";
          $data['description' . $i] = $postCondition->getDescription();
          $data['id_user' . $i] = $postCondition->getUser()->getId();
          $i++;
        }
        $sql = substr($sql, 0, -1);
        $stmt = $this->db->prepare($sql);
        $stmt->execute($data);
    Le résultat de la concaténation est le suivant :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    'INSERT INTO us_precondition (description, id_user) VALUES  (:description0, :id_user0), (:description1, :id_user1)'
    Si je le prends tel quel et que je remplace à la main les valeurs, que je tente dans phpmyadmin de 'insérer, ça marche.
    Mais PDO me répond "Fatal error: Uncaught PDOException: SQLSTATE[42000]: Syntax error or access violation: 1064 Erreur de syntaxe près de '('avec un tiret au début.', '41')' à la ligne 1", et précise que c'est la ligne $stmt->execute($data); qui lève l'erreur. Pourtant voici le debug :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
     
    execute( $params = ['description0' => 'Indiquer une ligne par condition,', 'id_user0' => 42, 'description1' => 'avec un tiret au debut.', 'id_user1' => 42] )
    J'ai également essayé de faire un entre deux :
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
     
        $sql = "INSERT INTO postcondition (description, id_user) VALUES (:description, :id_user)";
        $stmt = $this->db->prepare($sql);
        foreach ($postConditions as $postCondition) {
          $stmt->execute([
            'description' => $postCondition->getDescription(),
            'id_user' => $postCondition->getUser()->getId()
          ]);
        }
    Et cela fonctionne. Mais pour moi, cela revient au même que ma situation actuelle, c'est-à-dire que je fais plusieurs petits appels à la base de données, plutôt qu'une seul plus conséquent.

    Je ne comprends pas d'où vient le problème, on ne peut pas faire de requête multiple avec PDO ?

  2. #2
    Membre averti
    Homme Profil pro
    Étudiant
    Inscrit en
    Mai 2021
    Messages
    30
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2021
    Messages : 30
    Par défaut
    Hello,
    Bon ben ce matin, avec le même code, ça marche...
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
     
        $sql = "INSERT INTO postcondition (description, id_user) VALUES ";
        $i = 0;
        $data = [];
        foreach ($postConditions as $postCondition) {
          $sql .= "(:description" . $i . ", :id_user" . $i . "),";
          $data['description' . $i] = $postCondition->getDescription();
          $data['id_user' . $i] = $postCondition->getUser()->getId();
          $i++;
        }
        $sql = substr($sql, 0, -1);
        $stmt = $this->db->prepare($sql);
        $stmt->execute($data);
    J'aurais mieux fait de me coucher plus tôt !

  3. #3
    Expert confirmé Avatar de CosmoKnacki
    Homme Profil pro
    Justicier interdimensionnel
    Inscrit en
    Mars 2009
    Messages
    2 985
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Charente Maritime (Poitou Charente)

    Informations professionnelles :
    Activité : Justicier interdimensionnel

    Informations forums :
    Inscription : Mars 2009
    Messages : 2 985
    Par défaut
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    $postCondition->setId($this->db->lastInsertId());
    Méfie toi de lastInsertId lors d'une insertion multiple, il renvoie l'id du premier enregistrement de la dernière insertion (donc l'id du premier enregistrement).

  4. #4
    Membre averti
    Homme Profil pro
    Étudiant
    Inscrit en
    Mai 2021
    Messages
    30
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2021
    Messages : 30
    Par défaut
    Hello !
    Oui merci ! Dans ma première requête, c'était normal, parce que j'enregistrais les éléments un par un, donc ça allait de récupérer l'ID. Pour les autres, je l'ai enlevé.

  5. #5
    Membre averti
    Homme Profil pro
    Étudiant
    Inscrit en
    Mai 2021
    Messages
    30
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Isère (Rhône Alpes)

    Informations professionnelles :
    Activité : Étudiant

    Informations forums :
    Inscription : Mai 2021
    Messages : 30
    Par défaut
    Je viens de comprendre ce qui provoquait l'erreur : lorsque le tableau de données passé est vide. Et comme je ne vérifiais pas avant de lancer le execute s'il y avait bien des lignes dans le tableau, la requête SQL était pétée.

    C'est résolu !

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

Discussions similaires

  1. requete insert into qui ne fonctionne pas
    Par olivier252 dans le forum Débuter avec Java
    Réponses: 8
    Dernier message: 05/01/2020, 13h00
  2. [MySQL] update de cases à cocher multiples qui ne fonctionne pas
    Par mimosa21 dans le forum PHP & Base de données
    Réponses: 15
    Dernier message: 09/12/2012, 11h28
  3. [MySQL] INSERT INTO qui ne fonctionne pas mais qui fonctionne
    Par quiky dans le forum PHP & Base de données
    Réponses: 15
    Dernier message: 09/12/2009, 15h31
  4. Insert into qui ne fonctionne pas ?
    Par Little_flower dans le forum VBA Access
    Réponses: 3
    Dernier message: 05/08/2008, 10h12
  5. [MySQL] insertion étendue qui ne fonctionne pas
    Par osseili20 dans le forum PHP & Base de données
    Réponses: 2
    Dernier message: 01/02/2008, 10h39

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