前提条件

TransactionScope类需要引用System.Transactions;

数据库环境及需求

现在假设有两个表如图:

                                     

           表TA                                                                                  表TB

现有数据:                                            

现在的需求是:每往TA中插入一条数据,就更新TB的第一行,值为表TA的所有行的Age的平均值

可以看到表TB的Remark长度仅仅为2,待会利用这个制造错误

 

当不使用TransactionScope时:

 using (EFTestEntities db = new EFTestEntities())
{
//数据库TA原有2行,此时添加第3行
var aEntity = new TA
{
Name = "a",
Age =
};
db.TA.Add(aEntity); var listTA = db.TA.ToList();//此时list只有2行,新添加的没有读取到
int totalAge = ;
listTA.ForEach(p => totalAge += p.Age); //获取表TB的第一行,并修改AvgAge值
var bEntity = db.TB.First();
bEntity.AvgAge = totalAge / listTA.Count;
bEntity.Remark = "bb";
db.SaveChanges();
//结果是Tb的值没有变化
}

你认为是添加TA后没有db.SaveChanges();? 的确,添加这个之后,可以读取到新添加的值了,但是没有事务了,如:

using (EFTestEntities db = new EFTestEntities())
{
//数据库TA原有2行,此时添加第3行
var aEntity = new TA
{
Name = "a",
Age =
};
db.TA.Add(aEntity);
db.SaveChanges(); var listTA = db.TA.ToList();//此时list有3行了,新添加的可以被读取到
int totalAge = ;
listTA.ForEach(p => totalAge += p.Age); //获取表TB的第一行,并修改AvgAge值
var bEntity = db.TB.First();
bEntity.AvgAge = totalAge / listTA.Count;
bEntity.Remark = "bbc";//故意超出长度,制造错误
db.SaveChanges();
//结果是TA添加了新行,但是TB修改时出错导致修改失败。造成了数据不一致
}

结果是TA添加了新行,但是TB因为remark超出长度导致导致修改失败,此时TB的数据是错的。

怎样可以避免这个问题?答案就是用事务。

使用TransactionScope:

using (EFTestEntities db = new EFTestEntities())
{
using (TransactionScope ts = new TransactionScope())
{
//数据库TA原有2行,此时添加第3行
var aEntity = new TA
{
Name = "a",
Age =
};
db.TA.Add(aEntity);
db.SaveChanges();//重点,必须要db.SaveChanges(),然后下面才能获取到新添加的行 var listTA = db.TA.ToList();//此时list有3行了,新添加的可以被读取到
int totalAge = ;
listTA.ForEach(p => totalAge += p.Age); //获取表TB的第一行,并修改AvgAge值
var bEntity = db.TB.First();
bEntity.AvgAge = totalAge / listTA.Count;
bEntity.Remark = "bbc";//故意超出长度,制造错误
db.SaveChanges(); //执行到此处程序报错 ts.Complete();//事务提交未执行
//结果是自动回滚,相当于此次没有对数据库做任何操作。保持了数据一致性
}
}

下面给出正确的示例,实现以上需求:

using (EFTestEntities db = new EFTestEntities())
{
using (TransactionScope ts = new TransactionScope())
{
//数据库TA原有2行,此时添加第3行
var aEntity = new TA
{
Name = "a",
Age =
};
db.TA.Add(aEntity);
db.SaveChanges();//重点,必须要db.SaveChanges(),然后下面才能获取到新添加的行 var listTA = db.TA.ToList();//此时list有3行了,新添加的可以被读取到
int totalAge = ;
listTA.ForEach(p => totalAge += p.Age); //获取表TB的第一行,并修改AvgAge值
var bEntity = db.TB.First();
bEntity.AvgAge = totalAge / listTA.Count;
bEntity.Remark = "bb";//数据符合规范
db.SaveChanges(); //保存 ts.Complete();//提交事务
//结果是执行成功,TA多一条数据,同时TB的值也变了
}
}

一切都nice了!

EF中关于TransactionScope的使用的更多相关文章

  1. EF 中事务的书写

    在EF 中怎么使用事务? 这个问题纠结了我好久,直到有人跟我一起讨论,我和同事一起讨论查资料. 查的好多资料都是使用 using (TransactionScope scope = new Trans ...

  2. EF中的transaction的使用范例

    注意一点: 在EF中使用事物后,对于一个新增的model,在saveChanges后,可以得到该实体的自增ID,但在提交事物之前, 该数据并没有真正的新增到DB中,但此时可以得到model新增的自增I ...

  3. 在EF中正确的使用事务

    1.EF中使用事务: using (TransactionScope tran = new TransactionScope()) { try { using(var _context = new D ...

  4. 1.【使用EF Code-First方式和Fluent API来探讨EF中的关系】

    原文链接:http://www.c-sharpcorner.com/UploadFile/3d39b4/relationship-in-entity-framework-using-code-firs ...

  5. 2.EF中 Code-First 方式的数据库迁移

    原文链接:http://www.c-sharpcorner.com/UploadFile/3d39b4/code-first-migrations-with-entity-framework/ 系列目 ...

  6. EF中扩展出Between操作符 (修订版)

    随手记录一下,这是针对原文错误的修改. 原文:EF中扩展出Between操作符 直接使用是错误的,修改后的扩展方法: /// <summary> /// 扩展 Between 操作符 // ...

  7. 如何在EF中实现left join(左联接)查询

    在EF中,当在dbset使用join关联多表查询时,连接查询的表如果没有建立相应的外键关系时,EF生成的SQL语句是inner join(内联),对于inner join,有所了解的同学都知道,很多时 ...

  8. EF中执行sql语句,以及事务

    EF to sql string sql = "select T_Task.BSID,T_Task.CloseDate,T_Task.CompleteDate,T_Task.CloseUse ...

  9. EF架构~在ef中支持IQueryable级别的Contains被翻译成了Exists,性能可以接受!

    回到目录 Entityframeworks很聪明 不错,非常不错!ef里的contains比linq to sql里的contains有了明显的提升,事实上,是在进行SQL语句翻译上有所提升,在lin ...

随机推荐

  1. OC语言编写:为视图添加丝滑的水波纹

    先看一下最终效果图: 首先我们可以把如此丝滑的水波纹拆分一下下: 一条规律的曲线. 曲线匀速向右移动. 曲线下方的位置用颜色填充. 于是先来一条曲线吧. 对于需要产生波动如此规律的曲线,我们首先想到的 ...

  2. UNIX环境高级编程——守护进程

    一.守护进程简介 守护进程,也就是通常说的Daemon进程,是Linux中的后台服务进程.它是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件.守护进程常常在系 ...

  3. 多层界面之间显示与隐藏tabBar

    IOS中多层界面之间显示与隐藏tabBar? 在做项目的时候,遇到了一个难题,使用hidesBottomWhenPushed=YES属性设置,可以让本级界面及其以后界面都隐藏,但是根据项目 需求,在第 ...

  4. Chapter 1 Securing Your Server and Network(13):配置端点安全性

    原文出处:http://blog.csdn.net/dba_huangzj/article/details/38489765,专题目录:http://blog.csdn.net/dba_huangzj ...

  5. Socket编程实践(7) --Socket-Class封装(改进版v2)

    本篇博客定义一套用于TCP通信比较实用/好用Socket类库(运用C++封装的思想,将socket API尽量封装的好用与实用), 从开发出Socket库的第一个版本以来, 作者不知道做了多少改进,  ...

  6. mysql进阶(十二)常见错误汇总

    原因:外键名不能重复

  7. Leetcode_121_Best Time to Buy and Sell Stock

    本文是在学习中的总结,欢迎转载但请注明出处:http://blog.csdn.net/pistolove/article/details/43024967 Say you have an array ...

  8. 网站论坛同步用户,整合api,实现…

    在网上参考了很多资料后,终于完美实现了网站和discuz!nt论坛的双向整合,整合后网站和论坛之间可以同步注册.登录.退出和修改登录密码操作. 本系统的实现形式是新云CMS网站(ASP)和Discuz ...

  9. ADF BC New Features

      Examining ADF Business Components New Features Purpose In this tutorial, you create a series of si ...

  10. C语言之回文数算法

    "回文"是指正读反读都能读通的句子,它是古今中外都有的一种修辞方式和文字游戏,如"我为人人,人人为我"等.在数学中也有这样一类数字有这样的特征,成为回文数(pa ...