Entity Framework 6以前,框架本身并没有提供显式的事务处理方案,在EF6中提供了事务处理的API。

所有版本的EF,只要你调用SaveChanges方法进行插入、修改或删除,EF框架会自动将该操作进行事务包装。这种方法无法对事务进行显式的控制,例如新建事务等,可能会造成事务的粒度非常大,降低效率。EF不会对查询进行事务包装。

从EF6开始,默认情况下,如果每次调用Database.ExecuteSqlCommand(),如果其不在存在于任何事务中,则会将该Command包装到一个事务中。框架提供了多种重载,允许你重写这些方法,实现事务的控制。同样,执行存储过程的ObjectContext.ExecuteFunction()方法是实现了这种机制(但是ExecuteFunction不能被重写)。这两种情况下,使用的事务隔离级别均为数据库提供的默认隔离级别,SQL Server中使用的是READ COMMITED。

有同学提供了EF6之前版本的事务方案,如下:

 1 using (BlogDbContext context =new BlogDbContext())2 {3     using (TransactionScope transaction =new TransactionScope())4     {5         context.BlogPosts.Add(blogPost);6         context.SaveChanges();7         postBody.ID = blogPost.ID;8         context.EntryViewCounts.Add(9             new EntryViewCount() { EntryID = blogPost.ID });
10         context.PostBodys.Add(postBody);
11         context.SaveChanges();
12         //提交事务
13         transaction.Complete();
14     }
15 }

其实,上面方法执行结果不会错,但是存在隐患,这样情况下,显式事务其实是多余的。所以我对这种方案持怀疑态度(没有进行内部代码的分析,有时间了分析下,希望大家拍砖)。

官方体统的解决方案为:

 1 using System.Collections.Generic; 2 using System.Data.Entity; 3 using System.Data.SqlClient; 4 using System.Linq; 5 using System.Transactions; 6  7 namespace TransactionsExamples 8 { 9     class TransactionsExample
10     {
11         static void UsingTransactionScope()
12         {
13             using (var scope = new TransactionScope(TransactionScopeOption.Required))
14             {
15                 using (var conn = new SqlConnection("..."))
16                 {
17                     conn.Open();
18
19                     var sqlCommand = new SqlCommand();
20                     sqlCommand.Connection = conn;
21                     sqlCommand.CommandText =
22                         @"UPDATE Blogs SET Rating = 5" +
23                             " WHERE Name LIKE '%Entity Framework%'";
24                     sqlCommand.ExecuteNonQuery();
25
26                     using (var context =
27                         new BloggingContext(conn, contextOwnsConnection: false))
28                     {
29                         var query = context.Posts.Where(p => p.Blog.Rating > 5);
30                         foreach (var post in query)
31                         {
32                             post.Title += "[Cool Blog]";
33                         }
34                         context.SaveChanges();
35                     }
36                 }
37
38                 scope.Complete();
39             }
40         }
41     }
42 }

一般情况下,用户不需要对事务进行特殊的控制,使用EF框架默认行为即可。如果要对细节进行控制,参考下面章节:

EF6 API工作机制

EF6以前版本EF框架自己管理数据库连接,如果你自己尝试打开连接可能会抛出异常(打开一个已打开的连接会抛出异常)。由于事务必须在一个打开的连接上执行,因此要合并一系列操作到一个事务中,要么使用TractionScope,要么使用ObjectContext.Connection属性直接执行EntityConnection的Open(),并BeginTransaction()。另外,如果你在数据库底层连接上执行了事务,上面API会失败。

注意:EF6中移除了仅接受关闭连接的限制。

EF6 开始提供了:

Database.BeginTransaction() : 为用户提供一种简单易用的方案,在DbContext中启动并完成一个事务 -- 合并一系列操作到该事务中。同时使用户更方便的指定事务隔离级别。

Database.UseTransaction() : 允许DbContext使用一个EF框架外的事务。

在同一DbContext中合并一系列操作到一个事务中

Database.BeginTransaction()有两个重载方法。一个方法提供一个IsolationLevel参数,另一个无参方法使用底层数据库提供程序默认的数据库事务隔离级别。两个重载方法均返回一个DbContextTransaction对象,该对象提供Commit和Rollback方法,用于数据库底层事务的提交和回滚。

使用DbContextTransaction意味着,一旦提交或回滚事务,就要释放该对象。一种简单的方法是使用using语法,在using代码块结束时自动调用该对象的Dispose方法。

 1 using System; 2 using System.Collections.Generic; 3 using System.Data.Entity; 4 using System.Data.SqlClient; 5 using System.Linq; 6 using System.Transactions; 7  8 namespace TransactionsExamples 9 {
10     class TransactionsExample
11     {
12         static void StartOwnTransactionWithinContext()
13         {
14             using (var context = new BloggingContext())
15             {
16                 using (var dbContextTransaction = context.Database.BeginTransaction())
17                 {
18                     try
19                     {
20                         context.Database.ExecuteSqlCommand(
21                             @"UPDATE Blogs SET Rating = 5" +
22                                 " WHERE Name LIKE '%Entity Framework%'"
23                             );
24
25                         var query = context.Posts.Where(p => p.Blog.Rating >= 5);
26                         foreach (var post in query)
27                         {
28                             post.Title += "[Cool Blog]";
29                         }
30
31                         context.SaveChanges();
32
33                         dbContextTransaction.Commit();
34                     }
35                     catch (Exception)
36                     {
37                         dbContextTransaction.Rollback();
38                     }
39                 }
40             }
41         }
42     }
43 }

注意:启动一个事务需要底层数据库连接已打开。因此,如果连接未打开,调用Database.BeginTransaction()会打开连接,在其Dispose时关闭连接。

传递一个现有事务到DbContext

有时,你可能需要在同一数据库上,执行一个EF框架之外更大范围的事务,这是就需要自己打开连接并启动事务,然后通知EF框架:

1) 使用已打开的数据库连接

2) 在该连接上使用现有的事务

要实现上面的行为,你需要使用继承自DbContext的构造方法XXXContext(conn,contextOwnsConnection),其中:

conn : 是一个已存在的数据库连接

contextOwnsConnection : 是一个布尔值,指示上下文是否自己占用数据库连接。

注意:这种情况下,contextOwnsConnection必须设置为false,因为它通知EF框架,在自己使用完连接后,不要关闭它。见下面代码:

1 using (var conn = new SqlConnection("..."))
2 {
3     conn.Open();
4     using (var context = new BloggingContext(conn, contextOwnsConnection: false))
5     {
6     }
7 }

此外,你必须自己启动事务(如果你不想使用默认IsolationLevel,可以自己设置之)并让EF框架知道该连接上已经存在已启动的事务(参考下面代码的33行)。
      然后就可以直接在连接上执行数据库操作,或者在DbContext上执行,所有这些操作均在同一事务中执行,你负责提交或回滚事务,并调用DatabaseTransaction.Dispose(),最后要关闭和释放数据库连接。请参考以下代码:

 1 using System; 2 using System.Collections.Generic; 3 using System.Data.Entity; 4 using System.Data.SqlClient; 5 using System.Linq; 6 sing System.Transactions; 7  8 namespace TransactionsExamples 9 {
10      class TransactionsExample
11      {
12         static void UsingExternalTransaction()
13         {
14             using (var conn = new SqlConnection("..."))
15             {
16                conn.Open();
17
18                using (var sqlTxn = conn.BeginTransaction(System.Data.IsolationLevel.Snapshot))
19                {
20                    try
21                    {
22                        var sqlCommand = new SqlCommand();
23                        sqlCommand.Connection = conn;
24                        sqlCommand.Transaction = sqlTxn;
25                        sqlCommand.CommandText =
26                            @"UPDATE Blogs SET Rating = 5" +
27                             " WHERE Name LIKE '%Entity Framework%'";
28                        sqlCommand.ExecuteNonQuery();
29
30                        using (var context =
31                          new BloggingContext(conn, contextOwnsConnection: false))
32                         {
33                             context.Database.UseTransaction(sqlTxn);
34
35                             var query =  context.Posts.Where(p => p.Blog.Rating >= 5);
36                             foreach (var post in query)
37                             {
38                                 post.Title += "[Cool Blog]";
39                             }
40                            context.SaveChanges();
41                         }
42
43                         sqlTxn.Commit();
44                     }
45                     catch (Exception)
46                     {
47                         sqlTxn.Rollback();
48                     }
49                 }
50             }
51         }
52     }
53 }

注意:

  • 你可以传递null到方法Database.UseTransaction()来清除EF框架对当前事务的记忆。如果你这样做,事务既不会提交也不会回滚。所以要谨慎使用之,除非你确实需要这样。
  • 如果EF框架已经持有一个事务,此时你传递一个事务,Database.UseTransaction()将抛出一个异常:

★ EF框架已经持有一个事务;

★ 当EF框架已经在一个TransactionScope中运行;

★ 其数据库连接对象为null (例如,无连接--通常这种情况表示事务已经完成);

★ 数据库连接对象与EF框架的数据库连接对象不匹配;

对TransactionScope的一些补充

如果你使用.net framework 4.5.1及以上版本,可以使用TransactionScope的TransactionScopeAsyncFlowOption参数提供对异步的支持:

 1 using System.Collections.Generic; 2 using System.Data.Entity; 3 using System.Data.SqlClient; 4 using System.Linq; 5 using System.Transactions; 6  7 namespace TransactionsExamples 8 { 9     class TransactionsExample
10     {
11         public static void AsyncTransactionScope()
12         {
13             using (var scope = new TransactionScope(TransactionScopeAsyncFlowOption.Enabled))
14             {
15                 using (var conn = new SqlConnection("..."))
16                 {
17                     await conn.OpenAsync();
18
19                     var sqlCommand = new SqlCommand();
20                     sqlCommand.Connection = conn;
21                     sqlCommand.CommandText =
22                         @"UPDATE Blogs SET Rating = 5" +
23                             " WHERE Name LIKE '%Entity Framework%'";
24                     await sqlCommand.ExecuteNonQueryAsync();
25
26                     using (var context = new BloggingContext(conn, contextOwnsConnection: false))
27                     {
28                         var query = context.Posts.Where(p => p.Blog.Rating > 5);
29                         foreach (var post in query)
30                         {
31                             post.Title += "[Cool Blog]";
32                         }
33
34                         await context.SaveChangesAsync();
35                     }
36                 }
37             }
38         }
39     }
40 }

目前,使用TransactionScope还有一些限制:

  • 需要.NET 4.5.1及以上版本才支持异步方法;
  • 不能适用于云方案(除非你确保只有一个连接 -- 云方案不支持分布式事务);
  • 不能和Database.UseTransaction()结合使用;
  • 如果你的DDL代码存在问题(例如数据库初始化问题)或没有通过MSDTC服务来支持分布式事务,将抛出异常;

使用TransactionScope的优点:

  • 自动将本地事务升级为分布式事务:前提是你有不止一个连接到给定数据库或要组合一个连接到另一个数据库连接到同一事务(注意:你必须启动MSDTC服务以支持分布式事务)。
  • 易于编程。如果你更希望淡化对事务的关注,而非显示操作事务,使用TransactionScope将是一个更合适的选择。

随着EF6提供了Database.BeginTransaction()和Database.UseTransaction() 两个API,使用TransactionScope不在是必须的了。如果你依然使用TransactionScope,就必须留意上面限制。建议你尽可能使用新的API,而非TransactionScope。

了解Entity Framework中事务处理相关推荐

  1. 如何处理Entity Framework中的DbUpdateConcurrencyException异常

    如何处理Entity Framework中的DbUpdateConcurrencyException异常 参考文章: (1)如何处理Entity Framework中的DbUpdateConcurre ...

  2. 在Entity Framework中使用存储过程(一):实现存储过程的自动映射

    之前给自己放了一个比较长的假期,在这期间基本上没怎么来园子逛.很多朋友的留言也没有一一回复,在这里先向大家道个歉.最近一段时间的工作任务是如何将ADO.NET Entity Framework 4.0 ...

  3. Entity Framework中的Migrations

    Migrations是Entity Framework中非常有意思的一个工具.Migrations 的目的是用来跟踪数据库的改变. 假如我们想回滚到一个月前的代码,非常容易,有版本管理工具.但是要回滚 ...

  4. 关于Entity Framework中的Attached报错相关解决方案的总结

    关于Entity Framework中的Attached报错的问题,我这里分为以下几种类型,每种类型我都给出相应的解决方案,希望能给大家带来一些的帮助,当然作为读者的您如果觉得有不同的意见或更好的方法 ...

  5. 在Entity Framework中使用事务

    继续为想使用Entity Framework的朋友在前面探路,分享的东西虽然技术含量不高,但都是经过实践检验的. 在Entity Framework中使用事务很简单,将操作放在TransactionS ...

  6. Entity Framework中的Migration问题

    1.自从用上了Entity Framework(简称EF),妈妈再也不用担心我要写那么复杂的SQL语句了! 这是微软新一代的ORM工具,它能够将数据库的表中的记录映射成为程序中的一个对象,当然也能够将 ...

  7. Entity Framework中IQueryable, IEnumerable, IList的区别

    使用工具追踪EF生成的SQL 使用Entity Framework等ORM框架的时候,SQL对于使用者来说是透明的,往往很多人也不关心ORM所生成的SQL,然而系统出现性能问题的时候就必须关注生成的S ...

  8. Entity Framework中IQueryable, IEnumerable, IList的区别(转自网络)

    使用工具追踪EF生成的SQL 使用Entity Framework等ORM框架的时候,SQL对于使用者来说是透明的,往往很多人也不关心ORM所生成的SQL,然而系统出现性能问题的时候就必须关注生成的S ...

  9. 如何在 Entity Framework 中计算 时间差 ?

    咨询区 ison 我的项目中有一个需求,需要使用 Entity Framework 实现 日期差 的计算逻辑,参考如下代码: var now = DateTime.UtcNow;db.Items.Or ...

最新文章

  1. linux系统下设置oracle开机自动启动
  2. 洪小文博士写给你的新年书单
  3. 鸿蒙系统正式版官方下载,华为鸿蒙os2.0系统app正式版
  4. 若依微服务版怎样在common-core模块下引用第三方lib的jar包(MobileIMSDK4J_tcp的jar包)
  5. 【科大星云诗社动态20201204
  6. 离散数学实验题目-图
  7. boost::range_reverse_iterator相关的测试程序
  8. 计算机协会小游戏,网页闯关小游戏闯关记录(一)ISA TEST
  9. ApacheCN PythonWeb 译文集 20211028 更新
  10. JDK8新特性-Lambda表达式查找
  11. Java 图片处理解决方案:ImageMagick 快速入门教程
  12. backBarButtonItem 替换
  13. 《嵌入式Linux软硬件开发详解——基于S5PV210处理器》——2.3 SLC Nand Flash芯片
  14. 内存碎片产生原因及终极解决办法
  15. 计算机专业 在职跨英语,英语在职研究生跨专业可以吗?
  16. 2022-2028年全球与中国车辆传感器行业发展趋势及投资战略分析
  17. 如何在 iPhone 上设置整点报时提醒?
  18. 微信支付/退费(服务商)模式
  19. 伪静态、静态、动态url
  20. 散列算法和数字签名笔记

热门文章

  1. N皇后问题12 · N-Queens
  2. iOS开发技巧,细节(二)
  3. android 请求网络异步载入
  4. mooc_java 集合框架中 学生所选课程2MapHashMap
  5. 使用BIND安装智能DNS服务器(一)---基本的主从DNS服务器搭建
  6. 记一次 IIS 7.0 身份验证相关的问题解决
  7. 修改 MrBayes 3.2 源码解决不能恢复断点的问题
  8. mysql innodb redolog_MySQL · 引擎特性 · InnoDB redo log漫游(转)
  9. 收藏夹库计算机网络的关系,云南省计算机一级考试题库 计算机网络及基础.pdf...
  10. 1006 换个格式输出整数 (15 分)