设为首页 收藏本站
查看: 1720|回复: 0

[经验分享] sql server transaction

[复制链接]

尚未签到

发表于 2015-7-2 10:00:01 | 显示全部楼层 |阅读模式
  仔细研究了下,发现sql server里面的explicit transaction还是有点复杂的。以下是有些总结:

  ·         Commit transaction 会提交所有嵌套的transaction修改。但是如果嵌套的transaction里面有rollback tran to save point, 那么save point之后的部分会revert掉。
  delete from dbo.numbertable
  begin tran out1
       insert into dbo.numbertable values(1)
       insert into dbo.numbertable values(2)
       
       begin tran inn1
            insert into dbo.numbertable values(3)
            insert into dbo.numbertable values(4)
       save tran inn1SavePoint
            insert into dbo.numbertable values(5)
       rollback tran inn1SavePoint
       commit tran inn1
  commit tran out1
  
  ·         @@TRANCOUNT可以用来记录当前session transaction的个数,对于嵌套的transaction来讲,每次begin transaction都让它加一,每次commit tran都会让它减一。所以在语句里面可以通过select @@TRANCOUNT 来检查当前是否在一个transaction里面。如果当前@@TRANCOUNT为0,那调用commit还是rollback都会出现语句错误。在嵌套的transaction里面,rollback是很特殊的,它会直接把@@TRANCOUNT设置为0。
  begin tran
  begin tran
  begin tran
  print @@trancount
  rollback tran
  print @@trancount
  
  ·         对于嵌套的transaction来讲,rollback的写法是很特殊。如果嵌套,rollback transaction后面是不能带transaction的name的,要带也只能是最外面的transaction的name。Rollback只会抛弃所有嵌套transaction在rollback语句之前的修改。Rollback之后的更新依然提交就去了,原因在于:rollback之后,@@trancount为0,那么rollback之后的语句就不属于explicit transaction, 属于autocmmit transaction了,自动提交。
  delete from dbo.numbertable
  begin tran t1
       insert into dbo.numbertable values(1)
       
       begin tran t2
            insert into dbo.numbertable values(2)
       rollback tran
       print 'after rollback in innert transaction, the transaction count is: '+cast(@@trancount, varchar(5))
       insert into dbo.numbertable values(3)
  --commit tran
  select * from dbo.numbertable
  
  ·         存储过程里面也可以begin transaction,如果调用的地方也begin transaction,那么这种情况也属于嵌套transaction,如果在存储过程里面rollback,得到的结果和上面一样。但是有一点特殊的地方在与,执行存储过程结束的时候会比较开始执行sp的@@trancount和结束时候@@trancount的值,如果不一样,它会给出一个消息像“Transaction count after EXECUTE indicates that a COMMIT or ROLLBACK TRANSACTION statement is missing. Previous count = 1, current count = 0.”这个给出的消息并不会影响其后的执行。
  CREATE PROCEDURE [dbo].[AddNumber]        
  AS
  BEGIN
       begin tran
            insert into dbo.numbertable values(1)
            insert into dbo.numbertable values(2)
            insert into dbo.numbertable values(3)
       rollback tran
  END
  delete from dbo.numbertable
  begin tran out1
  exec dbo.addnumber
  print @@trancount
  insert into dbo.numbertable values(3)
  select * from dbo.numbertable
  
  ·         如果在sp里面rollback了,那到外满做commit, 或者rollback都是没有效果并且出错了,因为嵌套的transaction内部transaction一旦调用了rollback,@@trancount就为0了,在外满commit,rollback直接出错。比如如下sp,我想像在最外面rollback,那就出错了,因为sp里面语句rollback了。表里面始终会插入值3。
  delete from dbo.numbertable
  begin tran out1
  exec dbo.addnumber
  print @@trancount
  insert into dbo.numbertable values(3)
  rollback tran out1
  select * from dbo.numbertable
  
  ·         所有对于嵌套的transaction来讲,如果内部transaction一旦rollback,就会给外部的transaction留下一个大坑。为了解决这个为题,有两种解决方案:
  1.       在外部的transaction里面检查@@trancount,如果这个值跟你代码begin tran的可以一致,那说明内部transaction没有rollback,那可以继续commit或者rollback。
  delete from dbo.numbertable
  begin tran t1
       insert into dbo.numbertable values(1)
       
       begin transaction t2
            insert into dbo.numbertable values(2)
       rollback tran
  
       if @@trancount = 1
       begin
            insert into dbo.numbertable values(3)
            commit tran
       end
  2.       在所有的内部transaction里面,只能commit,不能rollback。如果必须rollback,那怎么办?save point就可以派上用场了。比如sp改成这样子:
  ALTER PROCEDURE [dbo].[AddNumber]         
  AS
  BEGIN
       begin tran
       save tran pp
            insert into dbo.numbertable values(1)
            insert into dbo.numbertable values(2)
            insert into dbo.numbertable values(3)
       rollback tran pp
       commit tran
  END
  
  begin tran out1
  exec dbo.addnumber
  print @@trancount
  insert into dbo.numbertable values(3)
  commit tran out1
  

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-82454-1-1.html 上篇帖子: 最近学习SQL SERVER,关于表的分区有点体会,请大家拍砖 下篇帖子: SQL Server 2008更改表“不允许保存更改”
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表