C#三十三 事务

简介: C#三十三 事务

  事务是一组组合成逻辑工作单元的数据库操作,在系统执行过程中可能会出错,但事务将控制和维护每个数据库的一致性和完整性。事务处理的主要特征是,任务要么全部完成,要么都不完成。在写入一些记录时,要么写入所有记录,要么什么都不写入。如果在写入一个记录时出现了一个失败,那么在事务处理中已写入的其他数据就会回滚。事务可能由很多单个任务构成。


       简单事务的一个常见例子:把钱从A账户转到B账户,这涉及两项任务,即从A账户把钱取出来;把钱存入B账户。两项任务要么同时成功,要么一起失败,给予回滚,以便保持账户的状态和原来相同。否则,在执行某一个操作的时候可能会因为停电、网络中断等原因而出现故障,所以有可能更新了一个表中的行,但没有更新相关表中的行。如果数据库支持事务,则可以将数据库操作组成一个事务,以防止因这些事件而使数据库出现不一致。


事务的ACID属性如下。

    原子性(Atomicity):事务的所有操作是原子工作单元;对于其数据修改,要么全都执行,要么全都不执行。原子性消除了系统处理操作子集的可能性。

       一致性(Consistency):数据从一种正确状态转换到另一种正确状态。事务在完成时,必须使所有的数据都保持一致。在相关数据库中,所有规则都必须应用于事务的修改,以保持所有数据的完整性。当事务结束时,所有的内部数据结构都必须是正确的。在存款取款的例子中,逻辑规则是,钱是不能凭空产生或销毁的,对于每个(收支)条目必须有一个相应的抵衡条目产生,以保证账户是平的。

      隔离性(Isolation):由并发事务所作的修改必须与任何其他并发事务所作的修改隔离。查看数据时数据所处的状态,要么是事务修改它之前的状态,要么是事务修改它之后的状态。简单的理解就是,防止多个并发更新彼此干扰。事务在操作数据时与其他事务操作隔离。隔离性一般是通过加锁的机制来实现的。

       持久性(Durability):事务完成之后,它对于系统的影响是永久性的。已提交的更改即使在发生故障时也依然存在。


                       对于事务的开发,.NET平台也为我们提供了几种非常简单方便的事务机制。无论是在功能上还是性能上都提供了优秀的企业级事务支持。


.NET开发者可以使用以下5种事务机制:


SQL和存储过程级别的事务。


ADO.NET级别的事务。


ASP.NET页面级别的事务。


企业级服务COM+事务。


System.Transactions事务处理。


这5种事务机制有着各自的优势和劣势,分别表现在性能、代码数量和部署设置等方面。开发人员可以根据项目的实际情况选择相应的事务机制。

SQL和存储过程级别的事务

       数据库事务是其他事务模型的基础,当一个事务创建时不同数据库系统都有自己的规则。SQL Server默认在自动提交的模式下工作,每个语句执行完后都会立即提交;与此对照的是Oracle需要你包含一个提交语句。但是当一个语句通过OLE DB执行时,它执行完后一个提交动作会被附加上去。例如:

1.     DECLARE @TranName VARCHAR(20); 
2.     SELECT @TranName = 'MyTransaction';  
3.      BEGIN TRANSACTION @TranName; 
5.     GO  
6.     USE AdventureWorks; 
7.     GO  
8.     DELETE FROM AdventureWorks.HumanResources.JobCandidate 
9.         WHERE JobCandidateID = 13; 
10. GO    
12. COMMIT TRANSACTION MyTransaction; 
13. GO  
14. 或者:  
15. CREATE PROCEDURE Tran1   
16. as    
17. begin tran   
18.         set xact_abort on  
19.         Insert Into P_Category(CategoryId,Name)values('1','test1')    
20.         Insert Into P_Category(CategoryId,Name)values('2','test2')    
21.         commit tran   
22. GO  
23. set xact_abort on表示遇到错误立即回滚。 
当然你也可以这么写:
1.     CREATE PROCEDURE tran1  
2.     as   
3.     begin tran  
4.     Insert Into P_Category(CategoryId,Name)values('1','test1')    
5.     if(@@error<>0)  
6.      rollback tran  
7.     else   
8.      begin   
9.       Insert Into P_Category(CategoryId,Name)values('2','test2')  
10.   if(@@error<>0)  
11.    rollback tran  
12.   else   
13.    commit tran  
14.  end   
15. GO 

数据库事务有它的优势和限制。

优势:

所有的事务逻辑包含在一个单独的调用中。

拥有运行一个事务的最佳性能。

独立于应用程序。

限制:

事务上下文仅存在于数据库调用中。

数据库代码与数据库系统有关。

10.3.3 ADO.NET级别的事务

现在我们对事务的概念和原理都有所了解了,并且作为已经有一些基础的C#开发者,我们已经熟知编写数据库交互程序的一些要点,即:


(1)使用SqlConnection类的对象的Open()方法建立与数据库服务器的连接。


(2)然后将该连接赋给SqlCommand对象的Connection属性。


(3)将欲执行的SQL语句赋给SqlCommand的CommandText属性。


(4)通过SqlCommand对象进行数据库操作。


       创建一个ADO.NET事务是很简单的,需要定义一个SqllTransaction类型的对象。SqlConnection 和OleDbConnection对象都有一个 BeginTransaction 方法,它可以返回 SqlTransaction 或者OleDbTransaction 对象。然后赋给SqlCommand对象的Transcation属性,即实现了二者的关联。为了使事务处理可以成功完成,必须调用SqlTransaction对象的Commit()方法。如果有错误,则必须调用Rollback()方法撤销所有的操作。


基于以上认识,下面我们就开始动手写一个基于ADO.NET的事务处理程序。

1.     string conString = "data source=127.0.0.1;database=codematic;user id=sa;  
2.     password=";  
3.     SqlConnection myConnection = new SqlConnection(conString); 
4.     myConnection.Open();       
6.     //启动一个事务 
7.     SqlTransaction myTrans = myConnection.BeginTransaction(); .      
9.     //为事务创建一个命令 
10. SqlCommand myCommand = new SqlCommand(); 
11. myCommand.Connection = myConnection; 
12. myCommand.Transaction = myTrans; 
13. try 
14. {  
15.     myCommand.CommandText = "update P_Product set Name='电脑2' where Id=52";  
16.     myCommand.ExecuteNonQuery(); 
17.     myCommand.CommandText = "update P_Product set Name='电脑3' where Id=53";  
18.     myCommand.ExecuteNonQuery(); 
19.     myTrans.Commit();//提交 
20.     Response.Write("两条数据更新成功");  
21. }  
22. catch (Exception ex) 
23. {  
24.     myTrans.Rollback();//遇到错误,回滚 
25.     Response.Write(ex.ToString());                
26. }  
27. finally 
28. {  
29.     myConnection.Close(); 
30. } 

ADO.NET事务的优势和限制如下。


优势:


简单。


和数据库事务差不多快。


事务可以跨越多个数据库访问。


独立于数据库,不同数据库的专有代码被隐藏了。


       限制:事务执行在数据库连接层上,所以需要在执行事务的过程中手动地维护一个连接。注意: 所有命令都必须关联在同一个连接实例上,ADO.NET事务处理不支持跨多个连接的事务处理。


视频课堂https://edu.csdn.net/course/play/7621


目录
相关文章
|
C# 数据库 消息中间件