Фиксация только определенных изменений, сделанных внутри транзакции, которые могут откатываться - PullRequest
5 голосов
/ 06 марта 2009

Это значительное изменение по сравнению с первоначальным вопросом, которое делает его более сжатым и охватывает вопросы, поднятые в существующих ответах ...

Можно ли сделать несколько изменений в нескольких таблицах внутри одной транзакции и откатить только некоторые изменения?

В TSQL ниже я бы не хотел, чтобы какие-либо изменения, сделанные "myLogSP", когда-либо были отменены. Но все изменения, сделанные различными myBusinessSP, должны откатываться при необходимости.

BEGIN TRANSACTION  

    EXEC myLogSP

    EXEC @err = myBusinessSPa
    IF (@err <> 0) BEGIN ROLLBACK TRANSACTION RETURN -1 END

    EXEC myLogSP

    EXEC @err = myBusinessSPb
    IF (@err <> 0) BEGIN ROLLBACK TRANSACTION RETURN -1 END

    EXEC myLogSP

    EXEC @err = myBusinessSPc
    IF (@err <> 0) BEGIN ROLLBACK TRANSACTION RETURN -1 END

    EXEC myLogSP

COMMIT TRANSACTION
RETURN 0

Порядок важен, myLogSP должны происходить между и после myBusinessSP (myLogSP воспринимают изменения, сделанные myBusinessSP)

Также важно, чтобы все myBusinessSP происходили внутри одной транзакции, чтобы сохранить целостность базы данных и разрешить откат всех их изменений при необходимости.

Это как если бы я хотел, чтобы myLogSP вели себя так, как будто они не являются частью транзакции. Это просто неудобный факт, что они оказываются внутри одного (из-за необходимости вызова между myBusinessSP.)

EDIT:

Окончательный ответ «нет», единственный вариант - изменить код. Либо использовать табличные переменные для ведения журнала (поскольку переменные не откатываются), либо изменить бизнес-логику на Не требовать транзакций ...

Ответы [ 7 ]

5 голосов
/ 06 марта 2009

Использование SAVEPOINT с , например,

BEGIN TRANSACTION  

    EXEC myLogSP

    SAVE TRANSACTION savepointA
    EXEC @err = myBusinessSPa
    IF (@err <> 0) BEGIN
        ROLLBACK TRANSACTION savepointA
        COMMIT
        RETURN -1
    END

    EXEC myLogSP

    SAVE TRANSACTION savepointB
    EXEC @err = myBusinessSPb
    IF (@err <> 0) BEGIN
        ROLLBACK TRANSACTION savepointB
        COMMIT
        RETURN -1
    END

    EXEC myLogSP

    SAVE TRANSACTION savepointC
    EXEC @err = myBusinessSPc
    IF (@err <> 0) BEGIN
        ROLLBACK TRANSACTION savepointC
        COMMIT
        RETURN -1
    END

    EXEC myLogSP

COMMIT TRANSACTION

EDIT

Исходя из предоставленной информации (и моего понимания), мне кажется, что вам придется заново создавать журналы SP, чтобы использовать переменные или файлы, или чтобы они запускались после факт "следующим образом:

BEGIN TRANSACTION  

    SAVE TRANSACTION savepointA
    EXEC @err = myBusinessSPa
    IF (@err <> 0) BEGIN
        ROLLBACK TRANSACTION savepointA
        EXEC myLogSPA -- the call to myBusinessSPa was attempted/failed
        COMMIT
        RETURN -1
    END

    SAVE TRANSACTION savepointB
    EXEC @err = myBusinessSPb
    IF (@err <> 0) BEGIN
        ROLLBACK TRANSACTION savepointB
        EXEC myLogSPA -- the call to myBusinessSPa originally succeeded
        EXEC myLogSPB -- the call to myBusinessSPb was attempted/failed
        COMMIT
        RETURN -1
    END

    SAVE TRANSACTION savepointC
    EXEC @err = myBusinessSPc
    IF (@err <> 0) BEGIN
        ROLLBACK TRANSACTION savepointC
        EXEC myLogSPA -- the call to myBusinessSPa originally succeeded
        EXEC myLogSPB -- the call to myBusinessSPb originally succeeded
        EXEC myLogSPC -- the call to myBusinessSPc was attempted/failed
        COMMIT
        RETURN -1
    END

    EXEC myLogSPA -- the call to myBusinessSPa succeeded
    EXEC myLogSPB -- the call to myBusinessSPb succeeded
    EXEC myLogSPC -- the call to myBusinessSPc succeeded

COMMIT TRANSACTION
2 голосов
/ 06 марта 2009

Вам нужно просто выйти за пределы текущего контекста. Есть несколько способов сделать это. Один (который я никогда не пробовал) - вызвать CLR для вставки.

Возможно, лучший способ - использовать тот факт, что на переменные таблицы не влияют транзакции. Например:

CREATE TABLE dbo.Test_Transactions
(
     my_string VARCHAR(20) NOT NULL
)
GO

DECLARE
     @tbl TABLE (my_string VARCHAR(20) NOT NULL)

BEGIN TRANSACTION

INSERT INTO dbo.Test_Transactions (my_string) VALUES ('test point one')

INSERT INTO @tbl (my_string) VALUES ('test point two')

INSERT INTO dbo.Test_Transactions (my_string) VALUES ('test point three')

ROLLBACK TRANSACTION

INSERT INTO dbo.Test_Transactions (my_string) select my_string from @tbl

SELECT * FROM dbo.Test_Transactions
SELECT * FROM @tbl
GO
2 голосов
/ 06 марта 2009

Нам посчастливилось поместить записи журнала в переменные таблицы и затем вставить их в реальные таблицы после фиксации или отката.

ОК, если вы не используете SQL Server 2008, попробуйте этот метод. Это грязный и обходной путь, но он должен работать. Таблица #temp и переменная таблицы должны быть установлены со структурой того, что возвращает sp.

create table #templog (fie1d1 int, field2 varchar(10))

declare @templog table (fie1d1 int, field2 varchar(10))

BEGIN TRANSACTION      
insert into #templog
Exec my_proc

insert into @templog (fie1d1, field2)
select t.* from #templog t 
left join @templog t2 on t.fie1d1 = t2.fie1d1 where t2.fie1d1 is null

insert into templog
values (1, 'test')

rollback tran
select * from #templog
select * from templog
select * from @templog
1 голос
/ 06 марта 2009
0 голосов
/ 06 марта 2009

Возможно, вы можете поместить вставки / обновления в бизнес-таблицы в их собственную атомарную транзакцию t1 и обернуть каждую из этих транзакций в другую транзакцию t2, которая выполняет обновление таблицы журнала, и t1 (обновления бизнес-таблицы) без каких-либо откатов. Например:

BEGIN TRANSACTION t2
     <insert to log>
     <execute stored procedure p1>
END TRANSACTION t2

CREATE PROCEDURE p1
AS
     BEGIN TRANSACTION t1
         <insert to business tables>
         <rollback t1 on error>
     END TRANSACTION t1

Я полагаю, что при откате t1 в хранимой процедуре вызов транзакции t2 не изменится.

0 голосов
/ 06 марта 2009

переместить инструкцию BEGIN TRANSACTION после первой вставки.

0 голосов
/ 06 марта 2009

Не проще ли перенести вставку журнала за пределы транзакции?

У меня нет ответа для блокировки таблицы, думаю, у вас уже есть ответ, будет иметь для блокировки таблицы, поскольку столбец идентификаторов может откатиться.

...