小分享:我有几张阿里云优惠券,用券购买或者升级阿里云相应产品最多可以优惠五折!领券地址:https://promotion.aliyun.com/ntms/act/ambassador/sharetouser.html?userCode=ohmepe03

< hr />

为了加深对EF特性的了解,so,写了一些测试代码。测试结果也许对实际项目没什么用处,但是对理解EF的相关机制还是有一定帮助的。本文可能会不定期更新(加入新的测试用例=。=)。

一、事务


直接看代码。

1、所有SaveChange包裹在一个TransactionScope里面。

 1 [TestMethod]
 2 public void TestMethod1()
 3 {
 4     using (var entities = new SysProcessEntities())
 5     {
 6         using (TransactionScope scope = new TransactionScope())
 7         {
 8             try
 9             {
10                 var test = entities.SysUser.Where(o => o.OrganizationID == 1).ToList();
11                 var test2 = entities.SysRole.ToList();
12
13                 test[0].Code = "admin1";
14                 test2[0].Name = "dddd";
15                 entities.SaveChanges();
16
17                 test[0].Code = "admin";
18                 test2[0].Name = "全功能";
19                 entities.SaveChanges();
20
21                 scope.Complete();
22             }
23             catch (Exception e)
24             {
25
26             }
27         }
28     }
29 }

结果:

2、在1的基础上去掉TransactionScope

 1 [TestMethod]
 2 public void TestMethod1()
 3 {
 4     using (var entities = new SysProcessEntities())
 5     {
 6         var test = entities.SysUser.Where(o => o.OrganizationID == 1).ToList();
 7         var test2 = entities.SysRole.ToList();
 8
 9         test[0].Code = "admin1";
10         test2[0].Name = "dddd";
11         entities.SaveChanges();
12
13         test[0].Code = "admin";
14         test2[0].Name = "全功能";
15         entities.SaveChanges();
16     }
17 }

结果:

3、注释掉2的第9行和第13行,检查单独的一条语句是否自带事务。

 1 [TestMethod]
 2 public void TestMethod1()
 3 {
 4     using (var entities = new SysProcessEntities())
 5     {
 6         var test = entities.SysUser.Where(o => o.OrganizationID == 1).ToList();
 7         var test2 = entities.SysRole.ToList();
 8
 9         //test[0].Code = "admin1";
10         test2[0].Name = "dddd";
11         entities.SaveChanges();
12
13         //test[0].Code = "admin";
14         test2[0].Name = "全功能";
15         entities.SaveChanges();
16     }
17 }

结果:

4、直接写Sql-"update [SysProcess].[dbo].[SysRole] set [name]='全功能' where ID=1",在查询分析器中执行,结果:

5、将1改为跨数据库

 1 [TestMethod]
 2 public void TestMethod1()
 3 {
 4     using (TransactionScope scope = new TransactionScope())
 5     {
 6         try
 7         {
 8             using (var entities = new SysProcessEntities())
 9             {
10                 var test = entities.SysUser.Where(o => o.OrganizationID == 1).ToList();
11                 var test2 = entities.SysRole.ToList();
12
13                 test[0].Code = "admin1";
14                 test2[0].Name = "dddd";
15                 entities.SaveChanges();
16
17                 test[0].Code = "admin";
18                 test2[0].Name = "全功能";
19                 entities.SaveChanges();
20             }
21             using (var entities = new DistributionEntities())
22             {
23                 var test = entities.VIPCard.ToList();
24                 test[0].Sex = true;
25                 entities.SaveChanges();
26             }
27
28             scope.Complete();
29         }
30         catch (Exception e)
31         {
32             string msg = e.Message;
33         }
34     }
35 }

结果:

于是整个世界美好了……

二、AsNoTracking


注意AsNoTracking要写在最终返回数据的那行代码中才有用,看代码:

 [TestMethod]
 public void TestMethod9()
 {
     using (var entities = new SysProcessEntities())
     {
         var ubs = entities.UserBrand.AsNoTracking();
         ).AsNoTracking();
         var brands = from ub in ubs
                         from ob in obs
                         where ub.BrandID == ob.BrandID
                         select ub.BrandID;
         var test = entities.ProBrand.Where(b => brands.Contains(b.ID)).ToList();
         Assert.AreEqual(EntityState.Detached, entities.Entry(test[]).State);
     }
 }

结果:

so,using之中应该这么写:

var ubs = entities.UserBrand;
);
var brands = from ub in ubs
                from ob in obs
                where ub.BrandID == ob.BrandID
                select ub.BrandID;
var test = entities.ProBrand.Where(b => brands.Contains(b.ID)).AsNoTracking().ToList();
Assert.AreEqual(EntityState.Detached, entities.Entry(test[]).State);

三、DbSet.Local


继续看代码:

1、

 public void TestMethod10()
 {
     var entities = new SysProcessEntities();
     var t1 = entities.ProBoduan.ToList();
     var t2 = entities.ProBoduan.ToList();
     Assert.AreEqual(t1[],t2[]);
 }

大家以为t1[0]是否等于t2[0]?答案是true。可是作为引用类型,我并没有重载它的Equals方法,照理应该为false才对呀。修改下测试代码:

 public void TestMethod10()
 {
     var entities = new SysProcessEntities();
     var t1 = entities.ProBoduan.ToList();
     var t2 = entities.ProBoduan.ToList();
     ].Name;
     t1[].Name = "随便取个名用来测试";
     Assert.AreEqual(name, t2[].Name);
 }

结果:。可知,t1[0]和t2[0]指向的是同一个对象,即t1和t2指向同一个数组地址,也就是entities.ProBoduan.Local指向的地址。不过数据库中,仍执行了两次取数操作。

2、then,加入AsNoTracking试试看:

public void TestMethod10()
{
    var entities = new SysProcessEntities();
    var t1 = entities.ProBoduan.AsNoTracking().ToList();
    var t2 = entities.ProBoduan.AsNoTracking().ToList();
    ].Name;
    t1[].Name = "随便取个名用来测试";
    Assert.AreEqual(name, t2[].Name); //true,t1的更改不影响t2,表明t1和t2指向不同地址
}

此时entities.ProBoduan.Local.Count为0。

3、

 public void TestMethod10()
 {
     var entities = new SysProcessEntities();
     var t2 = entities.ProBoduan.FirstOrDefault();

     var t1 = entities.ProBoduan.ToList();

 }

将第4行和第6行换下位置。

 public void TestMethod10()
 {
     var entities = new SysProcessEntities();
     var t1 = entities.ProBoduan.ToList();
     var count1 = entities.ProBoduan.Local.Count;//3
     var t2 = entities.ProBoduan.FirstOrDefault();

 }

四、AutoDetectChangesEnabled & ValidateOnSaveEnabled


 public void TestMethod9()
 {
     using (var entities = new SysProcessEntities())
     {
         entities.Configuration.AutoDetectChangesEnabled = false;
         var ubs = entities.UserBrand;
         );
         var brands = from ub in ubs
                         from ob in obs
                         where ub.BrandID == ob.BrandID
                         select ub.BrandID;
         var test = entities.ProBrand.Where(b => brands.Contains(b.ID)).ToList();
         ]).State;
         test[].Description = "ggggg";
         ]).State;
         entities.ProBrand.Remove(test[]);
     }
 }

t1、t2皆为EntityState.Unchanged,原因是entities.Configuration.AutoDetectChangesEnabled = false;若设为true,那么t2将为EntityState.Modified。注意此项设置对Added和Deleted没影响。当我们循环Add大批量数据到上下文中时,设为false将对性能有非常大的提升;该属性设置对循环修改已跟踪实体的属性(entity.Property = XXXX)关系不大,大批量修改已跟踪实体的属性的效率我测试过,非常快(不管是true还是false;当然设为false,然后修改实体属性,没什么意义)。

猜测:ValidateOnSaveEnabled表示在SaveChanges()时是否根据映射文件等判断实体是否符合规则(如Key是否被改变,默认已跟踪实体是不能手动改变Key值的等),以后测试!

转载请注明本文出处:http://www.cnblogs.com/newton/archive/2013/06/03/3115603.html

参考页面:.html

Entity Framework细节追踪的更多相关文章

  1. Asp.Net MVC 模型(使用Entity Framework创建模型类) - Part.1

    这篇教程的目的是解释在创建ASP.NET MVC应用程序时,如何使用Microsoft Entity Framework来创建数据访问类.这篇教程假设你事先对Microsoft Entity Fram ...

  2. Asp.Net MVC 模型(使用Entity Framework创建模型类)

    这篇教程的目的是解释在创建ASP.NET MVC应用程序时,如何使用Microsoft Entity Framework来创建数据访问类.这篇教程假设你事先对Microsoft Entity Fram ...

  3. 使用工具追踪Entity Framework生成的SQL

    学习entity framework期间收集的文章,转自http://www.cnblogs.com/hiteddy/archive/2011/10/01/Difference_among_IQuer ...

  4. 被Entity Framework Core的细节改进震撼了一下

    今天用 SQL Server Profiler 查看 Entity Framework Core 生成的 SQL 语句时,突然发现一个细节改进,并且被它震撼了一下: exec sp_executesq ...

  5. "Entity Framework数据插入性能追踪"读后总结

    园友莱布尼茨写了一篇<Entity Framework数据插入性能追踪>的文章,我感觉不错,至少他提出了问题,写了出来,引起了大家的讨论,这就是一个氛围.读完文章+评论,于是我自己也写了个 ...

  6. Entity Framework 6 Recipes 2nd Edition(13-5)译 -> 使POCO的修改追踪更高

    问题 你正在使用POCO,你想提高修改跟踪的性能,同时使内存消耗更少.另外,你想通过EF的CodeFirst方式来实现. 解决方案 假设你有一个关于Account(帐户)和相关的Payments(支付 ...

  7. Programming Entity Framework 翻译(1)-目录

    1. Introducing the ADO.NET Entity Framework ado.net entity framework 介绍 1 The Entity Relationship Mo ...

  8. Entity Framework Core 2.0 入门简介

    不多说废话了, 直接切入正题. EF Core支持情况 EF Core的数据库Providers: 此外还即将支持CosmosDB和 Oracle. EFCore 2.0新的东西: 查询: EF.Fu ...

  9. Entity Framework Core 2.0 入门

    该文章比较基础, 不多说废话了, 直接切入正题. 该文分以下几点: 创建Model和数据库 使用Model与数据库交互 查询和保存关联数据 EF Core支持情况 EF Core的数据库Provide ...

随机推荐

  1. linux下redis 集群配置

    redis.conf 配置文件说明 daemonize no --是否把redis-server启动在后台,默认是“否”.若改成yes pidfile /var/run/redis.pid --当Re ...

  2. angular : direative : scope | 指令scope里的符号@,=

    先看看以下的代码 <body ng-app="app" ng-controller="ctrl"> <dir myname="nam ...

  3. jwplayer 禁止视频的快进,但是可以后退(已实现)

    一直在研究.net 的视频播放,最近做起了jwplayer,然后项目要求是视频不能快进,但是可以重复观看已经看过的视频资源. 很简单 在标签<script> 中定义两个变量 var max ...

  4. Java Trie树

    Tire树,又叫字典树,主要是用来查找单词,词频统计的. 老规矩,直接上代码. package tireTree; public class TireTree { TireNode root; pub ...

  5. 咖啡师之路:第一日一杯Espresso

    代码敲累了.产品要发布了.熬夜啊加班啊. 精神完全不在状态. 咋办--- 咋办--- 咋办---! 来一杯Espresso浓缩咖啡.各位码农,码神们的必备良品! 咖啡每天要2-3杯,不管是速溶还是现磨 ...

  6. 一个好用的几乎没有Callback的Android异步库

    android-async-task 这是一个Android平台处理复杂异步任务的库 (https://github.com/gplibs/android-async-task) 1. 安装方法 gr ...

  7. wemall app商城源码android开发MD5加密工具类

    wemall-mobile是基于WeMall的android app商城,只需要在原商城目录下上传接口文件即可完成服务端的配置,客户端可定制修改.本文分享android开发MD5加密工具类主要代码,供 ...

  8. (0)写给Web初学者的教案-----Web基础

    0,Web基础 一.    What is the Web? Can It Eat? 很多同学可能都听说过一个名词叫做“Web”,这个词隐隐约约好像和我们上网相关.但是呢,又很难说的清楚.我们今天每位 ...

  9. Docker存储驱动之总览

    简介 本文会介绍Docker存储驱动的特性,别列出现在已经支持的存储驱动,最后,会介绍如果选型适合你的存储驱动. 可插拔的存储驱动架构 Docker的存储驱动架构是可插拔的,可以让你很方便的将适合你环 ...

  10. 爬取软考试题系列之ip自动代理

    马上5月份有个软件专业等级考试,以下简称软考,为了更好的复习备考,我打算抓取www.rkpass.com网上的软考试题. 以上为背景. 很久没有更新博客园的博客了,所以之前的代码没有及时的贴出来,咱们 ...