zoukankan      html  css  js  c++  java
  • C#中TransactionScope的使用方法和原理

    在.net 1.1的时代,还没有TransactionScope类,因此很多关于事务的处理,都交给了SqlTransaction和SqlConnection,每个Transaction是基于每个Connection的。这种设计对于跨越多个程序集或者多个方法的事务行为来说,不是非常好,需要把事务和数据库连接作为参数传入。

    在.net 2.0后,TransactionScope类的出现,大大的简化了事务的设计。示例代码如下:

    复制代码
    复制代码
    static void Main(string[] args)
            {
                using (TransactionScope ts = new TransactionScope())
                {
                    userBLL u = new userBLL();
                    TeacherBLL t = new TeacherBLL();
                    u.ADD();
                    t.ADD();
                    ts.Complete();
                }
            }
    复制代码
    复制代码

    只需要把需要事务包裹的逻辑块写在using (TransactionScope ts = new TransactionScope())中就可以了。从这种写法可以看出,TransactionScope实现了IDispose接口。除非显示调用ts.Complete()方法。否则,系统不会自动提交这个事务。如果在代码运行退出这个block后,还未调用Complete(),那么事务自动回滚了。在这个事务块中,u.ADD()方法和t.ADD()方法内部都没有用到任何事务类。

    TransactionScope是基于当前线程的,在当前线程中,调用Transaction.Current方法可以看到当前事务的信息。具体关于TransactionScope的使用方法,已经它的成员方法和属性,可以查看 MSDN 。

    TransactionScope类是可以嵌套使用,如果要嵌套使用,需要在嵌套事务块中指定TransactionScopeOption参数。默认的这个参数为Required。

    该参数的具体含义可以参考http://msdn.microsoft.com/zh-cn/library/system.transactions.transactionscopeoption(v=vs.80).aspx

    比如下面代码:

    复制代码
    复制代码
    static void Main(string[] args)
            {
                using (TransactionScope ts = new TransactionScope())
                {
                    Console.WriteLine(Transaction.Current.TransactionInformation.LocalIdentifier);
                    userBLL u = new userBLL();
                    TeacherBLL t = new TeacherBLL();
                    u.ADD();
                    using (TransactionScope ts2 = new TransactionScope(TransactionScopeOption.Required))
                    {
                        Console.WriteLine(Transaction.Current.TransactionInformation.LocalIdentifier);
                        t.ADD();
                        ts2.Complete();
                    }
                   ts.Complete();
                }
            }
    复制代码
    复制代码

    当嵌套类的TransactionScope的TransactionScopeOption为Required的时候,则可以看到如下结果,他们的事务的ID都是同一个。并且,只有当2个TransactionScope都complete的时候才能算真正成功。

    ED8FDA3E241D48B0A90F30A5AC8A9A59

    如果把TransactionScopeOption设为RequiresNew,则嵌套的事务块和外层的事务块各自独立,互不影响。

    复制代码
    复制代码
    static void Main(string[] args)
            {
                using (TransactionScope ts = new TransactionScope())
                {
                    Console.WriteLine(Transaction.Current.TransactionInformation.LocalIdentifier);
                    userBLL u = new userBLL();
                    TeacherBLL t = new TeacherBLL();
                    u.ADD();
                    using (TransactionScope ts2 = new TransactionScope(TransactionScopeOption.RequiresNew))
                    {
                        Console.WriteLine(Transaction.Current.TransactionInformation.LocalIdentifier);
                        t.ADD();
                        ts2.Complete();
                    }
                  ts.Complete();
                }
            }
    复制代码
    复制代码

    22D29B0134CD4993B629F085EEBF63D9

    可以看到,他们的事务id是不一样的。

    TransactionScopeOption设为Suppress则为取消当前区块的事务,一般很少使用。

    对于多个不同服务器之间的数据库操作,TransactionScope依赖DTC(Distributed Transaction Coordinator)服务完成事务一致性。

    但是对于单一服务器数据,TransactionScope的机制则比较复杂。主要用的的是线程静态特性。线程静态特性ThreadStaticAttribute让CLR知道,它标记的静态字段的存取是依赖当前线程,而独立于其他线程的。既然存储在线程静态字段中的数据只对存储该数据的同一线程中所运行的代码可见,那么,可使用此类字段将其他数据从一个方法传递到该第一个方法所调用的其他方法,而且完全不用担心其他线程会破坏它的工作。TransactionScope 会将当前的 Transaction 存储到线程静态字段中。当稍后实例化 SqlCommand 时(在此 TransactionScope 从线程局部存储中删除之前),该 SqlCommand 会检查线程静态字段以查找现有 Transaction,如果存在则列入该 Transaction 中。通过这种方式,TransactionScope 和 SqlCommand 能够协同工作,从而开发人员不必将 Transaction 显示传递给 SqlCommand 对象。实际上,TransactionScope 和 SqlCommand 所使用的机制非常复杂。具体可以参考文章 http://www.microsoft.com/china/MSDN/library/netFramework/netframework/NETMattersSep.mspx?mfr=true

    Wrox出版的《Professional C# 4 and .NET 4》也有关于TransactionScope的一些使用方法的介绍。

    原文转载自:https://www.cnblogs.com/sdaulldd/p/6022319.html,如有侵权,请联系博主删除。

  • 相关阅读:
    解决 el-table在ie11中默认状态下不是百分百展开的问题
    在vue中使用lexible.js 通过props传的style样式无法转换成rem
    解决 el-select 与 el-radio 组合字体发生闪烁问题
    解决element生成的下拉菜单,不在父级之下,而是在body之下,通过父级控制该下拉菜单的样式,达到不跟别的地方冲突的目的
    解决nvm use提示切换node版本成功,实际却没有切换过对应版本
    vue-element-admin安装失败的问题
    今天转正了,感谢博客园让我找到工作!
    electron-vue中点击按钮,实现打开程序目录里面的某个文件
    项目实战---模拟站酷网站
    vue实现选项卡切换效果
  • 原文地址:https://www.cnblogs.com/itjeff/p/12108981.html
Copyright © 2011-2022 走看看