Uso correcto de transacciones en SQL Server

236

Tengo 2 comandos y necesito que ambos se ejecuten correctamente o ninguno de ellos se ejecute. Así que creo que necesito una transacción, pero no sé cómo usarla correctamente.

¿Cuál es el problema con el siguiente 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

El INSERTcomando se ejecuta, pero el UPDATEcomando tiene un problema.

¿Cómo puedo implementar esto para revertir ambos comandos si alguno de ellos tiene un error en la ejecución?

Saeid
fuente

Respuestas:

513

Agregue un bloque try / catch, si la transacción tiene éxito, confirmará los cambios, si la transacción falla, la transacción se revierte:

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  
Darren
fuente
1
¿No se BEGIN TRANSACTION [Tran1]debe colocar dentro TRY? De todos modos, una pieza de código muy simple y elegante.
Piotr Nawrot
44
@PiotrNawrot No, si la creación de la transacción falló no hay necesidad de revertirla en la captura.
Monseñor
114

Al comienzo del procedimiento almacenado, se debe poner SET XACT_ABORT ON para indicar al servidor SQL que revierta automáticamente la transacción en caso de error. Si se omite o se establece en OFF, es necesario probar @@ ERROR después de cada declaración o usar el bloque de reversión TRY ... CATCH .

Nikola Markovinović
fuente
2
En otras palabras, su transacción no es atómica a menos que CONFIGURE XACT_ABORT ON primero.
4AM
Es difícil de ver con url subrayado, pero hay un guión bajo enXACT_ABORT
BurnsBA
32

Enfoque fácil:

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;
Bohdan
fuente