Uso corretto delle transazioni in SQL Server


236

Ho 2 comandi e ho bisogno che entrambi siano eseguiti correttamente o nessuno di essi sia eseguito. Quindi penso di aver bisogno di una transazione, ma non so come usarla correttamente.

Qual è il problema con il seguente script?

BEGIN TRANSACTION [Tran1]

INSERT INTO [Test].[dbo].[T1]
    ([Title], [AVG])
VALUES ('Tidd130', 130), ('Tidd230', 230)

UPDATE [Test].[dbo].[T1]
  SET [Title] = N'az2' ,[AVG] = 1
  WHERE [dbo].[T1].[Title] = N'az'

COMMIT TRANSACTION [Tran1]
GO

Il INSERTcomando viene eseguito, ma il UPDATEcomando ha un problema.

Come posso implementarlo per ripristinare entrambi i comandi se qualcuno di loro ha un errore nell'esecuzione?

Risposte:


513

Aggiungi un blocco try / catch, se la transazione ha esito positivo impegnerà le modifiche, se la transazione fallisce viene eseguito il rollback della transazione:

BEGIN TRANSACTION [Tran1]

  BEGIN TRY

      INSERT INTO [Test].[dbo].[T1] ([Title], [AVG])
      VALUES ('Tidd130', 130), ('Tidd230', 230)

      UPDATE [Test].[dbo].[T1]
      SET [Title] = N'az2' ,[AVG] = 1
      WHERE [dbo].[T1].[Title] = N'az'

      COMMIT TRANSACTION [Tran1]

  END TRY

  BEGIN CATCH

      ROLLBACK TRANSACTION [Tran1]

  END CATCH  

1
Non dovrebbe BEGIN TRANSACTION [Tran1]essere inserito all'interno TRY? Comunque - codice molto semplice ed elegante.
Piotr Nawrot,

4
@PiotrNawrot No, se la creazione della transazione non è riuscita non è necessario eseguirne il rollback nella cattura.
Mons.

114

All'inizio della procedura memorizzata si dovrebbe impostare SET XACT_ABORT ON per indicare al server SQL di eseguire automaticamente il rollback della transazione in caso di errore. Se ommited o impostato su OFF, è necessario testare @@ ERROR dopo ogni istruzione o utilizzare il blocco rollback TRY ... CATCH .


2
In altre parole, la tua transazione non è atomica a meno che non imposti prima XACT_ABORT ON.
4:00

È difficile vedere con la sottolineatura XACT_ABORT
dell'URL

32

Approccio semplice:

CREATE TABLE T
(
    C [nvarchar](100) NOT NULL UNIQUE,
);

SET XACT_ABORT ON -- Turns on rollback if T-SQL statement raises a run-time error.
SELECT * FROM T; -- Check before.
BEGIN TRAN
    INSERT INTO T VALUES ('A');
    INSERT INTO T VALUES ('B');
    INSERT INTO T VALUES ('B');
    INSERT INTO T VALUES ('C');
COMMIT TRAN
SELECT * FROM T; -- Check after.
DELETE T;
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.