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