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

SQLite Discussion :

UPDATE si différent sur valeur potentiellement NULL ?


Sujet :

SQLite

Vue hybride

Message précédent Message précédent   Message suivant Message suivant
  1. #1
    Membre chevronné
    Profil pro
    Inscrit en
    Octobre 2002
    Messages
    707
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2002
    Messages : 707
    Par défaut UPDATE si différent sur valeur potentiellement NULL ?
    Bonjour,

    Je souhaite mettre à jour certaines lignes de ma table *uniquement* si une certaine valeur a changé; le problème c'est que cette valeur peut être à NULL... et qu'on ne peut pas comparer avec NULL !
    Ex., soit la table...
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    CREATE TABLE personne
    	(prenom TEXT, nom TEXT, adresse TEXT, date_maj DATE);
     
    INSERT INTO personne
    VALUES
    	('Marcel', 'Picot', NULL, '2015-10-27'),
    	('Jules', 'Rambier', 'rue Gamma', '2015-10-27'),
    	('Leon', 'Citron', 'rue des Lilas', '2015-10-28');
    Je veux mettre à jour la première adresse uniquement si elle a changé, mais le code suivant de fonctionne pas car l'expression évaluée "NULL <> 'rue de la Chapelle" ne donne rien ! Alors que , évidemment, la nouvelle adresse diffère de l'ancienne qui était inconnue...

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    UPDATE personne
    SET adresse = 'rue de la Chapelle', date_maj = date('now')
    WHERE (adresse <> 'rue de la Chapelle') AND (prenom = 'Marcel');
     
    SELECT * FROM personne;
    Résultat:
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
     
    prenom	nom		adresse		date_maj
    Marcel	Picot 		(null)		2015-10-27		=> Au lieu de	Marcel	Picot 	rue de la Chapelle	2015-11-26
    Jules	Rambier		rue Gamma	2015-10-27
    Leon	Citron		rue des Lilas	2015-10-28
    De même dans l'autre sens, si la nouvelle adresse est "NULL", je dois pouvoir remplacer une ancienne adresse qui n'était pas à NULL.

    Pouvez-vous m'aider ?
    http://sqlfiddle.com/#!5/cf2fa/3

  2. #2
    Membre chevronné
    Profil pro
    Inscrit en
    Octobre 2002
    Messages
    707
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2002
    Messages : 707
    Par défaut
    PS: je ne veux pas faire d'UPDATE quand ce n'est pas nécessaire car je ne veux pas déclencher les triggers qui sont attachés à cet UPDATE.

  3. #3
    Membre émérite
    Homme Profil pro
    Chef de projets retraité
    Inscrit en
    Juillet 2011
    Messages
    458
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Cher (Centre)

    Informations professionnelles :
    Activité : Chef de projets retraité
    Secteur : Transports

    Informations forums :
    Inscription : Juillet 2011
    Messages : 458
    Par défaut
    Bonjour,

    Tu dois tester le NULL en même temps que la validation de ton adresse avec un opérateur OR soit :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    UPDATE personne
    SET adresse = 'rue de la Chapelle', date_maj = date('now')
    WHERE (adresse <> 'rue de la Chapelle' or adresse is null) AND (prenom = 'Marcel')
    Autre solution : utilise la fonction COALESCE

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    UPDATE personne
    SET adresse = 'rue de la Chapelle', date_maj = date('now')
    WHERE (coalesce(adresse, '') <> 'rue de la Chapelle' ) AND (prenom = 'Marcel')
    Pour annuler une adresse il te faudra une autre requête (ne positionner la colonne à NULL que si celle ci n'est pas NULL) soit :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    UPDATE personne
    SET adresse = null, date_maj = date('now')
    WHERE (adresse is not null) AND (prenom = 'Marcel')
    NB: Pour ce jeu d'essai le prénom est un discriminant de ligne suffisant. Dans la vie réelle je crais que cela ne suffise pas.

    Cordialement

  4. #4
    Membre chevronné
    Profil pro
    Inscrit en
    Octobre 2002
    Messages
    707
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2002
    Messages : 707
    Par défaut
    Citation Envoyé par acaumes Voir le message
    Bonjour,

    Tu dois tester le NULL en même temps que la validation de ton adresse avec un opérateur OR soit :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    UPDATE personne
    SET adresse = 'rue de la Chapelle', date_maj = date('now')
    WHERE (adresse <> 'rue de la Chapelle' or adresse is null) AND (prenom = 'Marcel')
    Merci pour ton aide.

    Le code ci-dessus force un update quand la nouvelle adresse est NULL et si l'ancienne est aussi NULL (donc pas différentes) !

    Autre solution : utilise la fonction COALESCE

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    UPDATE personne
    SET adresse = 'rue de la Chapelle', date_maj = date('now')
    WHERE (coalesce(adresse, '') <> 'rue de la Chapelle' ) AND (prenom = 'Marcel')
    Hmm, oui j'avais bien pensé à utiliser coalesce mais en fait ça ne résout qu'une partie du problème: dans le cas où la nouvelle adresse est '' il n'y aura pas d'update si le champ est à NULL ! De même si la nouvelle adresse est à NULL alors il y aura update même si l'ancienne est NULL !

    Pour annuler une adresse il te faudra une autre requête (ne positionner la colonne à NULL que si celle ci n'est pas NULL) soit :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    UPDATE personne
    SET adresse = null, date_maj = date('now')
    WHERE (adresse is not null) AND (prenom = 'Marcel')
    Je préfère utiliser une seule requête qui doit gérer tous les cas. Bon si vraiment il n' a pas d'autre solution, il faudra bien en passer par là mais ça m'embête :-\

    NB: Pour ce jeu d'essai le prénom est un discriminant de ligne suffisant. Dans la vie réelle je crais que cela ne suffise pas.

    Cordialement
    Oui comme tu as compris, c'est un exemple imaginaire, le "Marcel" c'est juste pour pouvoir tester le code sur la première ligne ;-)

  5. #5
    Membre émérite
    Homme Profil pro
    Chef de projets retraité
    Inscrit en
    Juillet 2011
    Messages
    458
    Détails du profil
    Informations personnelles :
    Sexe : Homme
    Localisation : France, Cher (Centre)

    Informations professionnelles :
    Activité : Chef de projets retraité
    Secteur : Transports

    Informations forums :
    Inscription : Juillet 2011
    Messages : 458
    Par défaut
    Bonjour,

    Pour faire le tout en une seul opération, il te faut utiliser une instruction CASE comme ceci :

    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    update personne
           SET adresse = 'rue de la Chapelle',
               date_maj = date('now')
    WHERE
         case
         when 'rue de la Chapelle' is null
              then adresse is not null
         else adresse <> 'rue de la Chapelle' or adresse is null
         end
         and prenom = 'Marcel'
    Ceci est un exemple. On doit si tu utilises un langage qui permet de travailler directement avec les fonctions de "binding" de SQLITE créer cette requête avec les varaibles idoines puis préparer la requête et utiliser les fonctions de binding décrites.

    Une autre solution c'est de créer une table temporaire que tu rempliras avec les données de mise à jour puis tu pourras utiliser les données de cette table avec une requête de ce type
    (J'ai créé cette table temporaire (npersonne) avec les mêmes noms de colonne que celles de la table principale)
    Code : Sélectionner tout - Visualiser dans une fenêtre à part
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    update personne
           SET adresse = (select adresse from NPersonne where prenom = personne.prenom),
               date_maj = date('now')
    WHERE
         case
         when (select adresse from NPersonne where prenom = personne.prenom) is null
              then adresse is not null
         else adresse <> (select adresse from NPersonne where prenom = personne.prenom) or adresse is null
         end
         and prenom in (select prenom from npersonne)

    Cordialement

  6. #6
    Membre chevronné
    Profil pro
    Inscrit en
    Octobre 2002
    Messages
    707
    Détails du profil
    Informations personnelles :
    Localisation : France

    Informations forums :
    Inscription : Octobre 2002
    Messages : 707
    Par défaut
    Je vais voir ce que je peux faire avec ça... Merci.

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

Discussions similaires

  1. [Débutant] LINQ to SQL, jointure avec une valeur potentiellement null
    Par gunth dans le forum Linq
    Réponses: 5
    Dernier message: 23/07/2013, 09h57
  2. UPDATE <table> SET <champ = valeur > IF NULL ?
    Par gelo91 dans le forum Requêtes
    Réponses: 5
    Dernier message: 04/03/2010, 16h28
  3. update qui marche pas sur valeur null
    Par touille dans le forum Requêtes
    Réponses: 2
    Dernier message: 16/07/2008, 12h05
  4. Requete sur valeur NULL
    Par LyLy_91 dans le forum Développement
    Réponses: 8
    Dernier message: 03/10/2007, 16h48
  5. Transact-SQL test sur valeur NULL
    Par samworld dans le forum Langage SQL
    Réponses: 3
    Dernier message: 20/08/2007, 23h11

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