问题求解,欢迎一起来讨论

king2003 2007-01-09 08:07:55
使用NORTHWIND数据库

CREATE PROCEDURE inner_sp @productid int AS

CREATE TABLE #temp (orderid int NOT NULL,
orderdate datetime NOT NULL)

PRINT 'This prints.'
BEGIN TRANSACTION

INSERT #temp (orderid, orderdate)
SELECT o.OrderID, o.OrderDate
FROM Orders
WHERE EXISTS (SELECT *
FROM [Order Details] od
WHERE od.OrderID = o.OrderID
AND od.ProductID = @productid)

COMMIT TRANSACTION
PRINT 'This does not print.'
go




CREATE PROCEDURE outer_sp AS

DECLARE @ret int
SET @ret = 4711
EXEC @ret = inner_sp 76

PRINT '@@error is ' + ltrim(str(@@error)) + '.'
PRINT '@@trancount is ' + ltrim(str(@@trancount)) + '.'
PRINT '@ret ' + coalesce(ltrim(str(@ret)), 'NULL') + '.'
IF @@trancount > 0 ROLLBACK TRANSACTION
go


EXEC outer_sp
go


This prints.
服务器: 消息 266,级别 16,状态 2,过程 inner_sp,行 22
EXECUTE 后的事务计数指出缺少了 COMMIT 或 ROLLBACK TRANSACTION 语句。原计数 = 0,当前计数 = 1。
服务器: 消息 107,级别 16,状态 1,过程 inner_sp,行 13
列前缀 'o' 与查询中所用的表名或别名不匹配。
服务器: 消息 107,级别 16,状态 1,过程 inner_sp,行 13
列前缀 'o' 与查询中所用的表名或别名不匹配。
服务器: 消息 107,级别 16,状态 1,过程 inner_sp,行 13
列前缀 'o' 与查询中所用的表名或别名不匹配。
@@error is 266.
@@trancount is 1.
服务器: 消息 266,级别 16,状态 2,过程 outer_sp,行 9
EXECUTE 后的事务计数指出缺少了 COMMIT 或 ROLLBACK TRANSACTION 语句。原计数 = 0,当前计数 = 1。
@ret 4711.
谁来解释一下
...全文
194 8 打赏 收藏 转发到动态 举报
AI 作业
写回复
用AI写文章
8 条回复
切换为时间正序
请发表友善的回复…
发表回复
king2003 2007-01-09
  • 打赏
  • 举报
回复
这几天老是在用过程,用的过程中对@@trancount感触挺多,愿和大家分享。今天也就在这儿献丑了,开一课讲讲@@trnacount.


  在用数据库的过程我不知道大家对这个消息是不是有印象:

服务器: 消息 266,级别 16,状态 2,过程 test,行 5
EXECUTE 后的事务计数指出缺少了 COMMIT 或 ROLLBACK TRANSACTION 语句。原计数 = 1,当前计数 = 0。”

  首先对这个消息解释一下:

错误 266
严重级别 16
消息正文
EXECUTE 后的事务计数指出缺少了 COMMIT 或 ROLLBACK TRANSACTION 语句。原计数 = %1!,当前计数 = %2!。

如果某存储过程退出时其 @@TRANCOUNT 值与进入该存储过程时不同,则 MS SQL Server 返回错误 266。

  有一点是要说明的:这个错误忽略,因为它只将消息发送到客户端而不影响执行。


  下面就用一个小例子来说明这个:
  首先:我们建一个TESTDB
create database testdb
go
use testdb
create proc p_1
as
begin tran dd
rollback tran dd
go

begin tran
exec p_1
rollback tran
go

按下F5 ,肯定会出现如下提示:

服务器: 消息 6401,级别 16,状态 1,过程 p_1,行 4
无法回滚 dd。没有找到任何该名称的事务或保存点。
服务器: 消息 266,级别 16,状态 1,过程 p_1,行 4
EXECUTE 后的事务计数指出缺少了 COMMIT 或 ROLLBACK TRANSACTION 语句。原计数 = 1,当前计数 = 2。


  这是我们在编写过程嵌套过程常会发生,因这个消息只是反馈给用户一个信息,大意的编程人员要是不在意思这个消息,会把一个错误的程序带给最终用户,相信其会发生什么事大家都知道,有时有的人知道了也苦于没有好的办法解决。我为什么要这么说,因为我前面已经说过,这个错误忽略,因为它只将消息发送到客户端而不影响执行。
  
  在这里我要说明,并不是说在一个过程中不能开始一个事务再结束一个事务,我要强调的只是要注意一下@@trancount入口 与出口的一致性。
  
  为了说清这一点,我必须再说明一个操作:Rollback, 不带 savepoint_name 和 transaction_name 的 ROLLBACK TRANSACTION 回滚到事务的起点。嵌套事务时,该语句将所有内层事务回滚到最远的 BEGIN TRANSACTION 语句。在这两种情况下,ROLLBACK TRANSACTION 均将 @@TRANCOUNT 系统函数减为 0。(关于rollback的操作说明很多,我再开新讲这个话题)
  
  有了上面的说明,我想大家应该知道我只要说明一点:当编写嵌套存储过程时要注意@@TRANSACTION 函数在存储过程的入口和出口处匹配,这样才能减少这个问题发生的可能性。
  
  有了问题,那就要解决,下面是我的方法,希望对大家有帮助。
  具体方法:
  1、在这样的过程中一般不要显式的开始一个事务,异常时再结束一个事务。
eg:
drop proc p_1
go
create proc p_1

as
..........
go


begin tran r
exec p_1
rollback tran r

  2、在嵌套的过程中不用rollback ,在@@trancount <> 0 时不Rollback,有点强人所难,是吧,实在要的啊,返回一个特定值给调用的过程 
A:在嵌套的过程中不用rollback
create porc p_2
as

begin tran
commit tran
go
create proc p_3
as
begin tran
exec p_2
commit tran
go

exec p_3
go

这个也不会失败

B:返回一个特定值给调用的过程
drop proc p_2
drop proc p_3

create proc p_2
as
if @@error <>0 return -1
go
create proc p_3
as
begin tran
declare @i int

exec @i = p_2
If @i = -1
rollback tran
else
commit tran
go

exec p_3
go


这一课到此为此,呵呵,提点意见!
king2003 2007-01-09
  • 打赏
  • 举报
回复
我明白了.看E文半天不明白.
如果某存储过程退出时其 @@TRANCOUNT 值与进入该存储过程时不同,则 MS SQL Server 返回错误 266
king2003 2007-01-09
  • 打赏
  • 举报
回复
我忘了加那话了,我是故意弄成那样的.就是不明白他提示这个的原因.当然了是第一句错误.
zheninchangjiang 2007-01-09
  • 打赏
  • 举报
回复
那系统的提示信息是正确的
zheninchangjiang 2007-01-09
  • 打赏
  • 举报
回复
先保证语句是正确的
king2003 2007-01-09
  • 打赏
  • 举报
回复
我就是故意弄成那样的.我想知道他出这种错误提示的原因.
marco08 2007-01-09
  • 打赏
  • 举报
回复
use northwind
go


CREATE PROCEDURE inner_sp @productid int AS

CREATE TABLE #temp (orderid int NOT NULL,
orderdate datetime NOT NULL)

PRINT 'This prints.'
BEGIN TRANSACTION

INSERT #temp (orderid, orderdate)
SELECT o.OrderID, o.OrderDate
FROM Orders o
WHERE EXISTS (SELECT *
FROM [Order Details] od
WHERE od.OrderID = o.OrderID
AND od.ProductID = @productid)

COMMIT TRANSACTION
PRINT 'This does not print.'
go


CREATE PROCEDURE outer_sp AS

DECLARE @ret int
SET @ret = 4711
EXEC @ret = inner_sp 76

PRINT '@@error is ' + ltrim(str(@@error)) + '.'
PRINT '@@trancount is ' + ltrim(str(@@trancount)) + '.'
PRINT '@ret ' + coalesce(ltrim(str(@ret)), 'NULL') + '.'
IF @@trancount > 0 ROLLBACK TRANSACTION
go


EXEC outer_sp
go

--result
/*
This prints.

(39 row(s) affected)

This does not print.
@@error is 0.
@@trancount is 0.
@ret 0.
*/
marco08 2007-01-09
  • 打赏
  • 举报
回复
--try

CREATE TABLE #temp (orderid int NOT NULL,
orderdate datetime NOT NULL)

PRINT 'This prints.'
BEGIN TRANSACTION

INSERT #temp (orderid, orderdate)
SELECT o.OrderID, o.OrderDate
FROM Orders o
WHERE EXISTS (SELECT *
FROM [Order Details] od
WHERE od.OrderID = o.OrderID
AND od.ProductID = @productid)

COMMIT TRANSACTION
PRINT 'This does not print.'
go

34,837

社区成员

发帖
与我相关
我的任务
社区描述
MS-SQL Server相关内容讨论专区
社区管理员
  • 基础类社区
  • 二月十六
  • 卖水果的net
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧