Uso correto de transações no SQL Server


236

Eu tenho 2 comandos e preciso que ambos sejam executados corretamente ou nenhum deles seja executado. Acho que preciso de uma transação, mas não sei como usá-la corretamente.

Qual é o problema com o seguinte 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

O INSERTcomando é executado, mas o UPDATEcomando tem um problema.

Como posso implementar isso para reverter os dois comandos, se algum deles tiver um erro na execução?

Respostas:


513

Adicione um bloco try / catch; se a transação for bem-sucedida, as alterações serão confirmadas; se a transação falhar, a transação será revertida:

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
Não deve BEGIN TRANSACTION [Tran1]ser colocado dentro TRY? Enfim - código muito simples e elegante.
Piotr Nawrot

4
@PiotrNawrot Não, se a criação da transação falhar, não há necessidade de recuperá-la na captura.
Monsenhor

114

No início do procedimento armazenado, deve-se colocar SET XACT_ABORT ON para instruir o Sql Server a reverter automaticamente a transação em caso de erro. Se omitido ou definido como OFF, é necessário testar @@ ERROR após cada instrução ou usar o bloco de reversão TRY ... CATCH .


2
Em outras palavras, sua transação não é atômica, a menos que você SET XACT_ABORT ON primeiro.
04:00

É difícil ver com o sublinhado do URL, mas há um sublinhado emXACT_ABORT
BurnsBA

32

Abordagem 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;
Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.