USE [TestDB]
GO
/****** 对象: Table [dbo].[Person] ******/
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‘,‘zhang‘)
Insert Into Person(PersonId,PersonName)
Values(‘1‘,‘wang‘)
Insert Into Person(PersonId,PersonName)
Values(‘3‘,‘sun‘)
Commit TransAction
/*
Select 一下 有‘1‘,‘zhang‘和‘3‘,‘sun‘,
说明只有第二句的错误被取消了
*/
问题原因:
“ XACT_ABORT ”开关用于指定当 Transact-SQL 语句出现运行时错误时,SQL Server 是否自动回滚到当前事务。当 SET XACT_ABORT 为 ON 时,如果执行 Transact-SQL 语句产生运行时错误,则整个事务将终止并回滚。当 SET XACT_ABORT 为 OFF 时,有时只回滚产生错误的 Transact-SQL 语句,而事务将继续进行处理。 如果错误很严重,那么即使 SET XACT_ABORT 为 OFF,也可能回滚整个事务。 OFF 是默认设置。编译错误(如语法错误)不受 SET XACT_ABORT 的影响。对于大多数 OLE DB 访问接口(包括 SQL Server),必须将隐式或显示事务中的数据修改语句中的 XACT_ABORT 设置为 ON。 唯一不需要该选项的情况是在提供程序支持嵌套事务时。
全部回滚的方法1:打开 XACT_ABORT
Use TestDB
SET XACT_ABORT ON -- 打开
Begin TransAction
Insert Into Person(PersonId,PersonName)
Values(‘1‘,‘zhang‘)
Insert Into Person(PersonId,PersonName)
Values(‘1‘,‘wang‘)
Insert Into Person(PersonId,PersonName)
Values(‘3‘,‘sun‘)
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‘,‘zhang‘)
Insert Into Person(PersonId,PersonName)
Values(‘1‘,‘wang‘)
Insert Into Person(PersonId,PersonName)
Values(‘3‘,‘sun‘)
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‘,‘zhang‘)
Set @tranError = @tranError + @@Error
Insert Into Person(PersonId,PersonName)
Values(‘1‘,‘wang‘)
Set @tranError = @tranError + @@Error
Insert Into Person(PersonId,PersonName)
Values(‘3‘,‘sun‘)
Set @tranError = @tranError + @@Error
If @tranError = 0
Commit TransAction
Else
Rollback TransAction
/*
自定义一个变量来判断最后是否发生过错误。
*/
特别注意:
如果一个事务写了 Begin TransAction 而没写 Commit TransAction 或 Rollback TransAction 则相关操作的数据会被锁住。而对于锁住的解决办法就是单独执行一下Commit TransAction 或 Rollback TransAction。
所以,在生产服务器上直接使用Sql语句时,用不带没写 Commit TransAction 或 Rollback TransAction 如下语句:
Begin TransAction
可操作的 Sql 语句
执行后,若操作结果正确,则单独执行Commit TransAction ;若操作结果不正确,则单独执行Rollback TransAction 。