zoukankan      html  css  js  c++  java
  • TransactionScope是什么

    TransactionScope使用说明

    TransactionScope是.Net Framework 2.0滞后,新增了一个名称空间。它的用途是为数据库访问提供了一个“轻量级”[区别于:SqlTransaction]的事物。使用之前必须添加对 System.Transactions.dll 的引用。

    本文导读:在实际开发工作中,执行一个事件,然后调用另一接口插入数据,如果处理逻辑出现异常,那么之前插入的数据将成为垃圾数据,我们所希望的是能够在整个这个方法定义为一个事务,TransactionScope 类提供一个简单方法,通过这一方法,您不必与事务本身交互,即可将代码块标记为参与某个事务。TransactionScope对象创建了一个事务,同时将该事务设置给Transaction类的Current属性。

    Transcope分布式事务 transaction事务如果是单阶段事务,并且由数据库直接处理,则属于本地事务

    一、TransactionScope的优点

    1、使用起来比较方便.TransactionScope可以实现隐式的事务,使你可以在写数据访问层代码的时候不用考虑到事务,而在业务层的控制事务.

    2、可以实现分布式事务,比如跨库或MSMQ. 

    二、TransactionScope缺点 

    1、性价比不高.比如,你只是在"Scope"里控制一个库的事务.用"TransactionScope"就有点浪费了.
    2、一般情况下只要你使用"TransactionScope",都要配置MSDTC,要配防火墙,要开139端口.这个端口不可以更改 

    三、如果你不得不用分布式事务,那也得琢磨琢磨 

    1.这步操作一定得在事务当中吗?这步操作如果没完成或者失败了,值得回滚整个事务吗?难道没有优雅的补偿措施或者容错措施? 

    2.分布式事务涉及到的点,必须的这么多?必须得实时的操作这一大串?不能通过通知类操作去精简掉某些点? 

    3.在发起分布式事务之后,你是不是做了事务无关的操作,尽管这些操作跟事务无关?(如,读取数据、计算、等用户返回消息、等其他模块的调用返回等等)要知道事务应该尽快结束。

    4.你没有把一些读操作也算在事务里面了吧?这是很容易犯的错误,你在事务中Enlist了一个select 操作。 

    5.你的操作,某些步骤可以等全部操作完成之后再执行.这类操作具有明显的通知类特点。通知类操作是说,我给你一个通知,并且我保证通知到了你;你必须吃下这个通知,并且保证处理成功,但是你不必我一通知你你就处理。这样的操作很明显可以用另外一个任务去搞。

    四、使用分布式事务注意如下几点

     1:确保参与事务的machine开启了分布式事务支持; 

    2:如果machine开启了防火墙,需要设置msdtc进程为例外; 

    3:参与事务的machine不能跨域(如果跨域,目前微软还没有确切的解决方案); 

    4:多数据库时才使用分布式事务,如果是同一个数据库,最好使用SqlTransaction.

    五、TransactionScope事务级别 

    在TransactionScope中默认的事务级别是Serializable,即在事务过程中,完全性锁表。别的进程不能查询,修改,新增,删除。这样会导致效率大大降低,虽然数据完整性很高。通常我们不需要那么高的数据完整性。所以需要修改默认的事务级别: 

    所有的事务级别如下

    Chaos 

    无法改写隔离级别更高的事务中的挂起的更改。

    ReadCommitted 

    不可以在事务期间读取可变数据,但是可以修改它。

    ReadUncommitted 

    可以在事务期间读取和修改可变数据。

    RepeatableRead 

    可以在事务期间读取可变数据,但是不可以修改。可以在事务期间添加新数据。

    Serializable

    可以在事务期间读取可变数据,但是不可以修改,也不可以添加任何新数据。

    Snapshot

    可以读取可变数据。在事务修改数据之前,它验证在它最初读取数据之后另一个事务是否更改过这些数据。如果数据已被更新,则会引发错误。这样使事务可获取先前提交的数据值。 

    在尝试提升以此隔离级别创建的事务时,将引发一个 InvalidOperationException,并产生错误信息“Transactions with IsolationLevel Snapshot cannot be promoted”(无法提升具有 IsolationLevel 快照的事务)。


    Unspecified 

    正在使用与指定隔离级别不同的隔离级别,但是无法确定该级别。如果设置了此值,则会引发异常。

    六、在C#中使用TransactionScope类(分布式事务),则须注意如下事项

    1、在项目中引用using System.Transactions命名空间(先要在添加net组件的引用);

    2、对MSDTC组件设置:

    在控制面板--->管理工具--->服务 中,开启Distributed Transaction Coordinator 服务。


    a、控制面板->管理工具->组件服务->计算机->我的电脑->右键->属性
    b、选择MSDTC页, 确认"使用本地协调器"
    c、点击下方"安全配置"按钮
     

    d、勾选: "允许网络DTC访问","允许远程客户端","允许入站","允许出站","不要求进行身份验证".
    e、对于数据库服务器端, 可选择"要求对呼叫方验证"
    f、勾选:"启用事务Internet协议(TIP)事务"。
    g、在双方防火墙中增加MSDTC.exe例外
    可用命令行: netsh firewall set allowedprogram %windir%/system32/msdtc.exe MSDTC enable

    3、重启IIS服务器。

    注意:

    我们只要确保数据库的打开操作是在事务范围内打开就行了。这样就可以做到事务的正确操作。

    如果WEB服务器和数据库是在同一台服务器上,TransactionScope使用的是本地事务,这时不需要配置MSDTC。

    如果WEB服务器和数据库不在同一台服务器上,TransactionScope会自动提升事务级别为分布式事务,这时就需要配置MSDTC。

    七、TransactionScope实例

     
    C# 代码   复制
            /**//// <summary>
            /// 发送消息
             /// </summary>
            /// <param name="sendUserId"></param>
            /// <param name="toUser">格式7FFA3AF2-E74B-4174-8403-5010C53E49A7|userName,7FFA3AF2-E74B-4174-8403-5010C53E49A7|userName</param>
            /// <param name="content"></param>
            /// <param name="sendedStatus">表示已送</param>
            /// <returns></returns>
            public static int sendMessage(string sendUserId, string toUser, string content, string sendedStatus)
            {           
                int receiveCount = 0;
                TransactionOptions transactionOption = new TransactionOptions();
    
                //设置事务隔离级别
                transactionOption.IsolationLevel = System.Transactions.IsolationLevel.ReadCommitted;
    
                // 设置事务超时时间为60秒
                transactionOption.Timeout = new TimeSpan(0, 0, 60);
    
                using (TransactionScope scope = new TransactionScope(TransactionScopeOption.Required, transactionOption))
                {
                    try
                    {
                        //在这里实现事务性工作
         //发送消息
                        insertMessage(sendUserId, toUser, content, sendedStatus);
    
         //在接收信息表中插入记录
                        receiveCount += insertReceiveMessage(userids[0], sendUserId, content, "0");
                       
                        // 没有错误,提交事务
                        scope.Complete();
                    }
                    catch (Exception ex) {
                        throw new Exception("发送信息异常,原因:"+ex.Message);
                    }finally{
                        //释放资源
                        scope.Dispose();
                      }                               
                }
                return receiveCount;
            }
    
  • 相关阅读:
    MyBatis参数传入集合之foreach动态sql
    【mybatis】多次查询缓存的问题
    svn is already locked解决方案
    Python 命令行之旅:使用 docopt 实现 git 命令
    Django 官方推荐的姿势:类视图
    这可能就是你苦苦寻找免费、高颜值、功能强大的 Markdown 编辑器(共5款)
    超级好用的 Java 数据可视化库:Tablesaw
    让你如“老”绅士般编写 Python 命令行工具的开源项目:docopt
    开发 Django 博客文章阅读量统计功能
    使用 Docker 让部署 Django 项目更加轻松
  • 原文地址:https://www.cnblogs.com/sandyliu1999/p/5098680.html
Copyright © 2011-2022 走看看