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