DÉSACTIVER ADBLOCK

ADBlock bloque du contenu sur le site

ADBlock errore
résultats trouvés: 

Question

J'ai une table de base de données et l'un des champs (pas la clé primaire) contient un index unique. Maintenant, je veux échanger les valeurs sous cette colonne pour deux lignes. Comment cela pourrait-il être fait? Je connais deux hacks:

  1. Supprimez les deux lignes et ré-insérez-les.
  2. Mettre à jour les lignes avec une autre valeur et permutez puis mettez à jour à la valeur réelle.

Mais je ne veux pas aller dans ce sens car ils ne semblent pas être la solution appropriée au problème. Quelqu'un pourrait-il m'aider?

Solution

Je pense que vous devriez choisir la solution 2. Il n'y a pas de fonction "swap" dans les variantes SQL que je connaisse.

Si vous devez le faire régulièrement, je suggère la solution 1, en fonction de la manière dont les autres parties du logiciel utilisent ces données. Vous pouvez avoir des problèmes de verrouillage si vous ne faites pas attention.

Mais en bref: il n’ya pas d’autre solution que celles que vous avez fournies.

Si vous aimez, laissez-nous votre avis

L'article a-t-il été utile et est-il traduit correctement?

AUTRES CONSEILS

Le mot magique est DEFERRABLE ici:

DROP TABLE ztable CASCADE;
CREATE TABLE ztable
    ( id integer NOT NULL PRIMARY KEY
    , payload varchar
    );
INSERT INTO ztable(id,payload) VALUES (1,'one' ), (2,'two' ), (3,'three' );
SELECT * FROM ztable;


    -- This works, because there is no constraint
UPDATE ztable t1
SET payload=t2.payload
FROM ztable t2
WHERE t1.id IN (2,3)
AND t2.id IN (2,3)
AND t1.id <> t2.id
    ;
SELECT * FROM ztable;

ALTER TABLE ztable ADD CONSTRAINT OMG_WTF UNIQUE (payload)
    DEFERRABLE INITIALLY DEFERRED
    ;

    -- This should also work, because the constraint 
    -- is deferred until "commit time"
UPDATE ztable t1
SET payload=t2.payload
FROM ztable t2
WHERE t1.id IN (2,3)
AND t2.id IN (2,3)
AND t1.id <> t2.id
    ;
SELECT * FROM ztable;

RÉSULTAT:

DROP TABLE
NOTICE:  CREATE TABLE / PRIMARY KEY will create implicit index "ztable_pkey" for table "ztable"
CREATE TABLE
INSERT 0 3
 id | payload
----+---------
  1 | one
  2 | two
  3 | three
(3 rows)

UPDATE 2
 id | payload
----+---------
  1 | one
  2 | three
  3 | two
(3 rows)

NOTICE:  ALTER TABLE / ADD UNIQUE will create implicit index "omg_wtf" for table "ztable"
ALTER TABLE
UPDATE 2
 id | payload
----+---------
  1 | one
  2 | two
  3 | three
(3 rows)

Suite à la réponse de Andy Irving

cela a fonctionné pour moi (sur SQL Server 2005) dans une situation similaire où j'ai une clé composite et que je dois permuter un champ qui fait partie de la contrainte unique.

clé: pID, LNUM rec1: 10, 0 rec2: 10, 1 rec3: 10, 2

et je dois échanger LNUM pour que le résultat soit

clé: pID, LNUM rec1: 10, 1 rec2: 10, 2 rec3: 10, 0

le code SQL requis:

UPDATE    DOCDATA    
SET       LNUM = CASE LNUM
              WHEN 0 THEN 1
              WHEN 1 THEN 2 
              WHEN 2 THEN 0 
          END
WHERE     (pID = 10) 
  AND     (LNUM IN (0, 1, 2))

Une autre approche fonctionne avec SQL Server: utilisez une jointure de table temporaire dans votre instruction UPDATE.

Le problème provient du fait que deux lignes ont la même valeur en même temps , mais si vous mettez à jour les deux lignes en même temps (avec leurs nouvelles valeurs uniques), il n'y a pas de violation de contrainte.

Pseudo-code:

-- setup initial data values:
insert into data_table(id, name) values(1, 'A')
insert into data_table(id, name) values(2, 'B')

-- create temp table that matches live table
select top 0 * into #tmp_data_table from data_table

-- insert records to be swapped
insert into #tmp_data_table(id, name) values(1, 'B')
insert into #tmp_data_table(id, name) values(2, 'A')

-- update both rows at once! No index violations!
update data_table set name = #tmp_data_table.name
from data_table join #tmp_data_table on (data_table.id = #tmp_data_table.id)

Merci à Rich H pour cette technique. - Mark

Je pense également que le n ° 2 est le meilleur choix, bien que je sois sûr de l'envelopper dans une transaction au cas où quelque chose se passerait mal à la mi-mise à jour.

Une alternative (puisque vous avez demandé) à la mise à jour des valeurs d'index unique avec des valeurs différentes serait de mettre à jour toutes les autres valeurs des lignes avec celles de l'autre ligne. Cela signifie que vous pouvez laisser les valeurs de l'index unique uniquement et que, finalement, vous obtenez les données souhaitées. Attention toutefois, si une autre table fait référence à cette table dans une relation de clé étrangère, toutes les relations de la base de données restent intactes.

J'ai le même problème. Voici mon approche proposée dans PostgreSQL. Dans mon cas, mon index unique est une valeur de séquence, définissant un ordre utilisateur explicite sur mes lignes. L'utilisateur mélange les lignes dans une application Web, puis soumet les modifications.

Je prévois d'ajouter un "avant". déclencheur. Dans ce déclencheur, chaque fois que ma valeur d'index unique est mise à jour, je vérifierai si une autre ligne contient déjà ma nouvelle valeur. Si tel est le cas, je leur donnerai mon ancienne valeur et la leur volera efficacement.

J'espère que PostgreSQL me permettra d'effectuer ce mélange aléatoire dans le déclencheur précédent.

Je vous répondrai et vous ferai savoir mon kilométrage.

En supposant que vous connaissiez la PK des deux lignes que vous souhaitez mettre à jour ... Cela fonctionne dans SQL Server, vous ne pouvez pas parler d'autres produits. SQL est (censé être) atomique au niveau de l'instruction:

CREATE TABLE testing
(
    cola int NOT NULL,
    colb CHAR(1) NOT NULL
);

CREATE UNIQUE INDEX UIX_testing_a ON testing(colb);

INSERT INTO testing VALUES (1, 'b');
INSERT INTO testing VALUES (2, 'a');

SELECT * FROM testing;

UPDATE testing
SET colb = CASE cola WHEN 1 THEN 'a'
                WHEN 2 THEN 'b'
                END
WHERE cola IN (1,2);

SELECT * FROM testing;

donc vous allez partir de:

cola    colb
------------
1       b
2       a

à:

cola    colb
------------
1       a
2       b

Pour Oracle, il existe une option, DEFERRED, mais vous devez l'ajouter à votre contrainte.

SET CONSTRAINT emp_no_fk_par DEFERRED; 

Pour reporter TOUTES les contraintes pouvant être différées pendant toute la session, vous pouvez utiliser l'instruction ALTER SESSION SET contraintes = DEFERRED.

Source

Oracle a différé la vérification d’intégrité qui résout ce problème, mais elle n’est disponible ni dans SQL Server ni dans MySQL.

Dans SQL Server, l'instruction MERGE peut mettre à jour des lignes qui rompraient normalement un UNIQUE KEY / INDEX. (Je viens de tester cela parce que j'étais curieux.)

Cependant, vous devrez utiliser une table / variable temporaire pour fournir MERGE avec les lignes nécessaires.

Je pense généralement à une valeur qu'aucun indice de ma table ne pourrait avoir. Habituellement, pour des valeurs de colonne uniques, c'est vraiment facile. Par exemple, pour les valeurs de la colonne 'position' (informations sur l'ordre de plusieurs éléments), la valeur est 0.

Ensuite, vous pouvez copier la valeur A dans une variable, la mettre à jour avec la valeur B, puis définir la valeur B à partir de votre variable. Deux questions, je ne connais pas de meilleure solution cependant.

1) changer les identifiants pour le nom

id    student 

1     Abbot   
2     Doris  
3     Emerson 
4     Green  
5     Jeames  

Pour l'exemple d'entrée, le résultat est:

id étudiant

1     Doris   
2     Abbot   
3     Green   
4     Emerson 
5     Jeames  

"Dans le cas où n nombre de lignes, comment allez-vous gérer ......"

Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow