码迷,mamicode.com
首页 > Web开发 > 详细

net中的4种事务总结

时间:2017-11-26 20:26:17      阅读:208      评论:0      收藏:0      [点我收藏+]

标签:方法调用   最新   block   comm   一个   tab   sql   sof   []   

在一个MIS系统中,没有用事务那就绝对是有问题的,要么就只有一种情况:你的系统实在是太小了,业务业务逻辑有只要一步执行就可以完成了。因此掌握事务处理的方法是很重要,进我的归类在.net中大致有以下4种事务处理的方法。大家可以参考一下,根据实际选择适当的事务处理。
1 SQL事务
    sql事务是使用SQL server自身的事务:在存储过程中直接使用Begin Tran,Rollback Tran,Commit Tran实现事务:
优点:执行效率最佳
限制:事务上下文仅在数据库中调用,难以实现复杂的业务逻辑。
Demo:(所有demo,都以SQL Server自带的Northwind数据的表Region为例)

技术分享图片
技术分享图片CREATE PROCEDURE dbo.SPTransaction
技术分享图片    (
技术分享图片    
@UpdateID int,
技术分享图片    
@UpdateValue nchar(50),
技术分享图片    
@InsertID int,
技术分享图片    
@InsertValue nchar(50)
技术分享图片    )
技术分享图片
AS
技术分享图片
begin Tran
技术分享图片
Update Region  Set RegionDescription=@UpdateValue where RegionID=@UpdateID
技术分享图片
技术分享图片
insert into Region Values (@InsertID,@InsertValue)
技术分享图片
技术分享图片
declare @RegionError int
技术分享图片
select @RegionError=@@error
技术分享图片
if(@RegionError=0)
技术分享图片
COMMIT Tran
技术分享图片
else
技术分享图片
ROLLBACK Tran
技术分享图片
GO
技术分享图片
技术分享图片执行带事务的存储过程


2 ADO.net事务
   Ado.net事务可能是大家一般都用的
 优点:简单,效率和数据库事务差不多。
 缺点:事务不能跨数据库,只能在一个数据库连接上。如果是两个数据库上就不能使用该事务了。
Demo:

技术分享图片
技术分享图片/// <summary>
技术分享图片        
/// 一般的ADO.net 事务
技术分享图片        
/// </summary>

技术分享图片        public void ADONetTran1()
技术分享图片        
{
技术分享图片            SqlConnection conn 
= new SqlConnection("Data Source=127.0.0.1;Initial Catalog=Northwind;Persist Security Info=True;User ID=sa;Password=123;");
技术分享图片            SqlCommand cmd 
= new SqlCommand();
技术分享图片            
try
技术分享图片            
{
技术分享图片                cmd.CommandText 
= "Update Region Set RegionDescription=@UpdateValue where RegionID=@UpdateID";
技术分享图片                cmd.CommandType 
= CommandType.Text;
技术分享图片                cmd.Connection 
= conn;
技术分享图片                conn.Open();
技术分享图片                SqlParameter[] paras 
= new SqlParameter[]{
技术分享图片                                        
new SqlParameter ("@UpdateID",SqlDbType.Int,32),
技术分享图片                                        
new SqlParameter ("@UpdateValue",SqlDbType .NChar,50)}
;
技术分享图片                paras[
0].Value = "2";
技术分享图片                paras[
1].Value = "Update Value12";
技术分享图片
技术分享图片                
foreach (SqlParameter para in paras)
技术分享图片                
{
技术分享图片                    cmd.Parameters.Add(para);
技术分享图片                }

技术分享图片                
//开始事务
技术分享图片
                cmd.Transaction = conn.BeginTransaction();
技术分享图片                cmd.ExecuteNonQuery();
技术分享图片
技术分享图片
技术分享图片                cmd.CommandText 
= "insert into Region values(@InsertID,@InsertValue)";
技术分享图片                cmd.CommandType 
= CommandType.Text;
技术分享图片
技术分享图片                paras 
= new SqlParameter[]{
技术分享图片                                        
new SqlParameter ("@InsertID",SqlDbType.Int ,32),
技术分享图片                                        
new SqlParameter ("@InsertValue",SqlDbType.NChar ,50)}
;
技术分享图片                paras[
0].Value = "7";
技术分享图片                paras[
1].Value = "Insert Value";
技术分享图片
技术分享图片                cmd.Parameters.Clear();
技术分享图片                
foreach (SqlParameter para in paras)
技术分享图片                
{
技术分享图片                    cmd.Parameters.Add(para);
技术分享图片                }

技术分享图片                
技术分享图片                cmd.ExecuteNonQuery();
技术分享图片                
//提交事务
技术分享图片
                cmd.Transaction.Commit();
技术分享图片            }

技术分享图片            
catch
技术分享图片            
{
技术分享图片                
//回滚事务
技术分享图片
                cmd.Transaction.Rollback();
技术分享图片                
throw;
技术分享图片            }

技术分享图片            
finally
技术分享图片            
{
技术分享图片                conn.Close();
技术分享图片            }

技术分享图片
技术分享图片        }

3 TransactionScope事务
  TransactionScope事务类,它可以使代码块成为事务性代码。并自动提升为分布式事务
 优点:实现简单,同时能够自动提升为分布式事务
Demo:

技术分享图片
技术分享图片 /// <summary>
技术分享图片        
/// TransactionScope事务:可自动提升事务为完全分布式事务的轻型(本地)事务。 
技术分享图片        
/// 使用时要保证MSDTC服务(控制分布事务)是开启的可以使用:net start msdtc命令开启服务;
技术分享图片        
/// </summary>

技术分享图片        public void ADONetTran2()
技术分享图片        
{
技术分享图片             SqlConnection conn 
= new SqlConnection("Data Source=127.0.0.1;Initial Catalog=Northwind;Persist Security Info=True;User ID=sa;Password=123;");
技术分享图片             SqlCommand cmd 
= new SqlCommand();
技术分享图片            
try
技术分享图片            
{
技术分享图片              
技术分享图片                
using (System.Transactions.TransactionScope ts = new TransactionScope())
技术分享图片                
{
技术分享图片                    
技术分享图片                    cmd.CommandText 
= "Update Region Set RegionDescription=@UpdateValue where RegionID=@UpdateID";
技术分享图片                    cmd.CommandType 
= CommandType.Text;
技术分享图片                    cmd.Connection 
= conn;
技术分享图片                    conn.Open();
技术分享图片                    SqlParameter[] paras 
= new SqlParameter[]{
技术分享图片                                        
new SqlParameter ("@UpdateID",SqlDbType.Int,32),
技术分享图片                                        
new SqlParameter ("@UpdateValue",SqlDbType .NChar,50)}
;
技术分享图片                    paras[
0].Value = "2";
技术分享图片                    paras[
1].Value = "Update Value12";
技术分享图片
技术分享图片                    
foreach (SqlParameter para in paras)
技术分享图片                    
{
技术分享图片                        cmd.Parameters.Add(para);
技术分享图片                    }

技术分享图片                    cmd.ExecuteNonQuery();
技术分享图片
技术分享图片
技术分享图片                    cmd.CommandText 
= "insert into Region values(@InsertID,@InsertValue)";
技术分享图片                    cmd.CommandType 
= CommandType.Text;
技术分享图片
技术分享图片                    paras 
= new SqlParameter[]{
技术分享图片                                        
new SqlParameter ("@InsertID",SqlDbType.Int ,32),
技术分享图片                                        
new SqlParameter ("@InsertValue",SqlDbType.NChar ,50)}
;
技术分享图片                    paras[
0].Value = "8";
技术分享图片                    paras[
1].Value = "Insert Value";
技术分享图片
技术分享图片                    cmd.Parameters.Clear();
技术分享图片                    
foreach (SqlParameter para in paras)
技术分享图片                    
{
技术分享图片                        cmd.Parameters.Add(para);
技术分享图片                    }

技术分享图片
技术分享图片                    cmd.ExecuteNonQuery();
技术分享图片                    
//提交事务
技术分享图片
                    ts.Complete();
技术分享图片                }

技术分享图片            }

技术分享图片            
catch
技术分享图片            
{
技术分享图片                
throw;
技术分享图片            }

技术分享图片            
finally
技术分享图片            
{
技术分享图片                conn.Close();
技术分享图片            }

技术分享图片
技术分享图片        }

4 COM+事务
  在分布式应用程序中,往往要同时操作多个数据库,使用数据库事务就不能满足业务的要求了。在COM+中,提供完整的事务处理服务。很方便处理多个数据库上的事务。
Demo:

技术分享图片
技术分享图片/// <summary>
技术分享图片        
/// COM+事务
技术分享图片        
/// </summary>

技术分享图片        public void ComTran()
技术分享图片        
{
技术分享图片            SqlConnection conn 
= new SqlConnection("Data Source=127.0.0.1;Initial Catalog=Northwind;Persist Security Info=True;User ID=sa;Password=123;");
技术分享图片            SqlCommand cmd 
= new SqlCommand();
技术分享图片            ServiceConfig sc 
= new ServiceConfig();
技术分享图片
技术分享图片            
//指定事务类型
技术分享图片
            sc.Transaction = TransactionOption.Required;
技术分享图片            
//设置启动跟踪
技术分享图片
            sc.TrackingEnabled = true;
技术分享图片            
//创建一个上下文,该上下文的配置由作为 cfg 参数传递的 ServiceConfig 对象来指定。
技术分享图片            
//随后,客户端和服务器端的策略均被触发,如同发生了一个方法调用。
技术分享图片            
//接着,新的上下文被推至上下文堆栈,成为当前上下文
技术分享图片
            ServiceDomain.Enter(sc);
技术分享图片            
try
技术分享图片            
{
技术分享图片                    cmd.CommandText 
= "Update Region Set RegionDescription=@UpdateValue where RegionID=@UpdateID";
技术分享图片                    cmd.CommandType 
= CommandType.Text;
技术分享图片                    cmd.Connection 
= conn;
技术分享图片                    conn.Open();
技术分享图片                    SqlParameter[] paras 
= new SqlParameter[]{
技术分享图片                                        
new SqlParameter ("@UpdateID",SqlDbType.Int,32),
技术分享图片                                        
new SqlParameter ("@UpdateValue",SqlDbType .NChar,50)}
;
技术分享图片                    paras[
0].Value = "2";
技术分享图片                    paras[
1].Value = "Update Value22";
技术分享图片
技术分享图片                    
foreach (SqlParameter para in paras)
技术分享图片                    
{
技术分享图片                        cmd.Parameters.Add(para);
技术分享图片                    }

技术分享图片                    cmd.ExecuteNonQuery();
技术分享图片
技术分享图片
技术分享图片                    cmd.CommandText 
= "insert into Region values(@InsertID,@InsertValue)";
技术分享图片                    cmd.CommandType 
= CommandType.Text;
技术分享图片
技术分享图片                    paras 
= new SqlParameter[]{
技术分享图片                                        
new SqlParameter ("@InsertID",SqlDbType.Int ,32),
技术分享图片                                        
new SqlParameter ("@InsertValue",SqlDbType.NChar ,50)}
;
技术分享图片                    paras[
0].Value = "9";
技术分享图片                    paras[
1].Value = "Insert Value";
技术分享图片
技术分享图片                    cmd.Parameters.Clear();
技术分享图片                    
foreach (SqlParameter para in paras)
技术分享图片                    
{
技术分享图片                        cmd.Parameters.Add(para);
技术分享图片                    }

技术分享图片
技术分享图片                    cmd.ExecuteNonQuery();
技术分享图片
技术分享图片                    
//提交事务
技术分享图片
                    ContextUtil.SetComplete();
技术分享图片            }

技术分享图片            
catch
技术分享图片            
{
技术分享图片                
//回滚事务
技术分享图片
                ContextUtil.SetAbort();
技术分享图片                
throw;
技术分享图片            }

技术分享图片            
finally
技术分享图片            
{
技术分享图片                conn.Close();
技术分享图片                
//触发服务器端的策略,随后触发客户端的策略,如同一个方法调用正在返回。
技术分享图片                
//然后,当前上下文被弹出上下文堆栈,调用 Enter 时正在运行的上下文成为当前的上下文。
技术分享图片
                ServiceDomain.Leave();
技术分享图片            }

技术分享图片
技术分享图片        }

在.net中还有些也能进行事务处理,如web Service中

需要特别补充的是:
如果你使用的是分布事务(TransactionScope事务和COM+事务),在默认情况下你是要重新配置安装SQL Server数据库服务器和访问数据库的客户端的.(如果没有配置运行会出现以下错误:该伙伴事务管理器已经禁止了它对远程/网络事务的支持。 (异常来自 HRESULT:0x8004D025)
)下面是MSDN上关于配置分布式事务的一段原话:
配置分布式事务
要启用分布式事务,可能需要通过网络启用 MS DTC,以便在使用应用了最新的 Service Pack 的较新操作系统(例如 Windows XP 或 Windows 2003)时使用分布式事务。如果启用了 Windows 防火墙(Windows XP Service Pack 2 的默认设置),必须允许 MS DTC 服务使用网络或打开 MS DTC 端口。
实际怎么配置呢,经过我的实际使用:大致如下:打开‘控制面板‘->‘管理工具‘->‘组件服务‘,点开‘组件服务‘->‘计算机‘->‘我的电脑‘,在‘我的电脑‘上右击属性,点‘MSDTC‘,然后点‘安全性配置‘。作为数据库的服务器的配置如下:
技术分享图片

而访问数据库的客户端的配置和服务器端的稍有些差别:

技术分享图片
在设置完上面的还有使防火墙MS DTC 服务使用网络或打开 MS DTC 端口:运行netsh firewall set allowedprogram %windir%/system32/msdtc.exe MSDTC enable命令就可以了

ASP.NET中的自动化事务 

    通过在ASP.NET页面中添加Transaction属性,可使得ASP.NET能够在系统中支持自动事务。利用Transaction属性,开发人员能够指示页面参与现有事务,开始新事务,或者不参与事务。下表列举了ASP.NET中可用的Transaction属性值。 

    通过在代码中的Page指令中设置Transaction属性能够定义页面支持的事务级别。例如,插入以下指令能够保证页面活动总是在事务范围中执行: 

<%@ Page Transaction="Required" %>

    如果省略Transaction属性,页面则禁用事务。使用System.EnterpriseServices.ContextUtil类的静态方法在ASP.NET页面中提交或者放弃事务。这些静态方法是SetComplete()和SetAbort()(它们分别对应Page事件CommitTransaction()和AbortTransaction())。以下代码列举了页面实现框架,该页面将Page指令的Transaction属性设置为Required,同时在CommitTransaction()和AbortTransaction()事件中,编写处理事务结果所需的代码。

  void Page_Load(object sender, System.EventArgs e)
   {
       AbortTransaction += new System.EventHandler(AbortTransactionEvent);
       CommitTransaction += new System.EventHandler(CommitTransactionEvent);
       try
        {
             /* 在这里放置事务性代码 */
             ContextUtil.SetComplete();
         }
        catch (Exception)
         {
             ContextUtil.SetAbort();
         }
        }
        void AbortTransactionEvent(object sender, System.EventArgs e)
        {
            /*用于回滚行为的代码*/
        }
        void CommitTransactionEvent(object sender, System.EventArgs e)
        {
            /*用于提交行为的代码*/
        } 
}

4. 何时使用事务 

    虽然.NET 2.0对事务提供了很好的支持,但是没有必要总是使用事务。使用事务的第一条规则是,在能够使用事务的时候都应该使用事务,但是不要使用过度。原因在于,每次使用事务,都会占用一定的开销。另外,事务可能会锁定一些表的行。还有一条规则是,只有当操作需要的时候才使用事务。例如,如果只是从数据库中查询一些记录,或者执行单个查询,在大部分时候都不需要使用显式事务。 

    开发人员应该在头脑中始终保持一个概念,就是用于修改多个不同表数据的冗长事务会严重妨碍系统中的所有其他用户。这很可能导致一些性能问题。当实现一个事务时,遵循下面的实践经验能够达到可接受的结果:(1)避免使用在事务中的SELECT返回数据,除非语句依赖于返回数据;(2)如果使用SELECT语句,只选择需要的行,这样不会锁定过多的资源,而尽可能的提高性能;(3)尽量将事务全部写在T-SQL或者API中;(4)避免事务与多重独立的批处理工作结合,应该将这些批处理放置在单独的事务中;(5)尽可能避免大量更新。 
另外,必须注意的一点就是事务的默认行为。在默认情况下,如果没有显式的提交事务,则事务会回滚。虽然默认行为允许事务的回滚,但是显式回滚方法总是一个良好的编程习惯。这不仅仅只是释放锁定数据,也将使得代码更容易读并且更少错误。 

 

net中的4种事务总结

标签:方法调用   最新   block   comm   一个   tab   sql   sof   []   

原文地址:http://www.cnblogs.com/lgx5/p/7899929.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!