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

简单事务的一个常见例子:把钱从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.      

4.     BEGIN TRANSACTION @TranName; 

5.     GO  

6.     USE AdventureWorks; 

7.     GO  

8.     DELETE FROM AdventureWorks.HumanResources.JobCandidate 

9.         WHERE JobCandidateID = 13; 

10. GO  

11.  

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事务是很简单的,需要定义一个SqlTransaction类型的对象。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(); 

5.      

6.     //启动一个事务 

7.     SqlTransaction myTrans = myConnection.BeginTransaction(); 

8.      

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事务处理不支持跨多个连接的事务处理。