淡淡回忆 发表于 2015-7-3 11:18:32

sql server中嵌套事务

  我们在写事务时经常遇到的问题如下:



消息 266,级别 16,状态 2,过程 sp1,第 0 行
EXECUTE 后的事务计数指示 BEGIN 和 COMMIT 语句的数目不匹配。上一计数 = 1,当前计数 = 0。
消息 3903,级别 16,状态 1,过程 sp2,第 15 行
ROLLBACK TRANSACTION 请求没有对应的 BEGIN TRANSACTION。

  如果这只是一个单独的事务引起的,那么很好解决,我们只要检查下是否遗漏了匹配的BEGIN tran 和 COMMIT tran即可,但是如果2个存储过程都是用事务写的,那么就即使每个存储过程的事务写法都正常,也会报这个错误,
  这是因为只要子事务里有回滚语句:如ROLLBACK      那么全局的@@TRANCOUNT被直接置为0了,导致父事务提交时发现 @@TRANCOUNT=0报错 ,sql server会认为当前不存在任何事务,在父存储过程中任何的COMMIT TRAN或
  ROLLBACK 语句都会找不到它对应的 BEGIN TRAN   
  下面我们用一个实例来看下:
  假设有一张表,ID为非自增主键



USE
GO
/****** Object:Table .    Script Date: 02/17/2013 15:44:35 ******/
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
SET ANSI_PADDING ON
GO
CREATE TABLE .(
NOT NULL,
NULL,
(50) NULL,
CONSTRAINT PRIMARY KEY CLUSTERED
(
ASC
)WITH (PAD_INDEX= OFF, STATISTICS_NORECOMPUTE= OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS= ON, ALLOW_PAGE_LOCKS= ON) ON
) ON
GO
SET ANSI_PADDING OFF
GO
  
  我们常规的写一个插入的子存储过程如下:



USE
GO
/****** Object:StoredProcedure .    Script Date: 02/17/2013 15:46:46 ******/
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
--内层事务存储过程,演示如何处理才能在嵌套的事务存储过程中正确处理事务
ALTER PROCEDURE .
@ID BIGINT ,
@UserID BIGINT ,
@Name VARCHAR(50)
AS
BEGIN
SET XACT_ABORT ON   
BEGIN TRAN   
IF(EXISTS(SELECT TOP 1 * FROM dbo.test WHERE ID=@ID))   
BEGIN
ROLLBACK      
RETURN 0 ;
END
--业务逻辑开始
INSERTdbo.test
( ID, UserID, Name)
VALUES( @ID,
@UserID,
@Name
)
--业务逻辑结束
IF @@error0
BEGIN
ROLLBACK                                 
RETURN 0;
END      
COMMIT   
     SET XACT_ABORT OFF;   
     RETURN 1 ;
END
  调用的父存储过程如下:



USE
GO
/****** Object:StoredProcedure .    Script Date: 02/17/2013 16:09:09 ******/
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
-- =============================================
-- Author:      
-- Create date:
-- Description:   
-- =============================================
ALTER PROCEDURE .
@ID BIGINT ,
@UserID BIGINT ,
@Name VARCHAR(50)
AS
BEGIN TRAN   
DECLARE @result INT
EXEC @result = innertranv1 @ID =@ID, @UserID =@UserID, @Name = @Name
IF ( @result
页: [1]
查看完整版本: sql server中嵌套事务