欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

SQL Server 中的回滚

程序员文章站 2022-12-24 11:13:49
默认情况下如果执行一个事务中出现错误,则只回滚错误操作语句(就是说这句不执行了,算不上回滚),错误处之前或之后的正确操作语句还是会被提交。如: 全部回滚的方法1:打开 XACT_ABORT 全部回滚方法2:使用Try...Catch 全部回滚方法3:自定义错误变量 最后要注意的是:如果一个事务写了  ......
use [testdb]
go
/****** 对象:  table [dbo].[person]    脚本日期: 11/23/2008 13:37:48 ******/
set ansi_nulls on
go
set quoted_identifier on
go
create table [dbo].[person](
    [personid] [nchar](18) not null,
    [personname] [nchar](20) not null,
 constraint [pk_person] primary key clustered 
(
    [personid] asc
)with (pad_index  = off, statistics_norecompute  = off, ignore_dup_key = off, allow_row_locks  = on, allow_page_locks  = on) on [primary]
) on [primary]

默认情况下如果执行一个事务中出现错误,则只回滚错误操作语句(就是说这句不执行了,算不上回滚),错误处之前或之后的正确操作语句还是会被提交。如:

use testdb

begin transaction
    insert into person(personid,personname)
                values('1','name1')
    insert into person(personid,personname)
                values('1','name1')
    insert into person(personid,personname)
                values('3','name3')
commit transaction
/*
    select 一下 有'1','name1'和'3','name3',
    说明只有第二句的错误被取消了
*/

全部回滚的方法1:打开 xact_abort

use testdb
set xact_abort on -- 打开
begin transaction
    insert into person(personid,personname)
                values('1','name1')
    insert into person(personid,personname)
                values('1','name1')
    insert into person(personid,personname)
                values('3','name3')
commit transaction
/*
    当 set xact_abort 为 on 时,
    如果执行 transact-sql 语句产生运行时错误,
    则整个事务将终止并回滚。 
    默认情况下它是off状态。
*/

全部回滚方法2:使用try...catch

use testdb
begin try
    begin transaction
        insert into person(personid,personname)
                    values('1','name1')
        insert into person(personid,personname)
                    values('1','name1')
        insert into person(personid,personname)
                    values('3','name3')
    commit transaction
end try
begin catch
    rollback transaction
end catch
/*
    使用trycatch来捕获异常。
    如果 try 块内生成的错误导致当前事务的状态失效,
    则将该事务归类为不可提交的事务。
    如果通常在 try 块外中止事务的错误在 try 内发生时,
    就会导致事务进入不可提交状态。
    不可提交的事务只能执行读操作或 rollback transaction。
    该事务不能执行任何可能生成写操作或 commit transaction 的 transact-sql 语句。
    如果事务被分类为不可提交的事务,则 xact_state 函数会返回值 -1。
*/

全部回滚方法3:自定义错误变量

use testdb
declare @tranerror int -- 定义变量
set @tranerror=0
    begin transaction
        insert into person(personid,personname)
                    values('1','name1')
            set @tranerror = @tranerror + @@error
        insert into person(personid,personname)
                    values('1','name1')
            set @tranerror = @tranerror + @@error
        insert into person(personid,personname)
                    values('3','name3')
            set @tranerror = @tranerror + @@error
    if @tranerror = 0
        commit transaction
    else
        rollback transaction
/*
    自定义一个变量来判断最后是否发生过错误。
*/

最后要注意的是:如果一个事务写了 begin transaction 而没写 commit transaction 或 rollback transaction 则相关操作的数据(也许是表,也许是列,这我还没测试。。。)会被锁住。。。而对于锁住的解决办法就是单独执行一下commit transaction 或 rollback transaction