问题

你想有效地获取只是用来显示不会更新的操作的实体.另外,你想用CodeFirst的方式来实现

解决方案

一个非常常见行为,尤其是网站,就是只是让用户浏览数据.大多数情况下,用户不会更新数据.在这种情况下,你可以通过避开上下文的缓存和修改跟踪来提高代码性能,你可以非常简单地使用AsNoTracking方法来实现.

让我们假设你一个应用程序来管理doctor(医生)的appointments(预约),你的模型如下图Figure 13-5.

Figure 13-5. A model for managing doctors and their appointments

 

         首先,这个例子用EFCodeFirst方式来实现.,Listing 13-6创建我们的实体类, Company, Doctor, 和Appointment.

Listing 13-6. The Company, Doctor,and Appointment.Entity Object

public class Company

{

public Company()

{

Doctors = new HashSet<Doctor>();

}

public int CompanyId { get; set; }

public string Name { get; set; }

public virtual ICollection<Doctor> Doctors { get; set; }

}

public class Doctor

{

public Doctor()

{

Appointments = new HashSet<Appointment>();

}

public int DoctorId { get; set; }

public string Name { get; set; }

public int CompanyId { get; set; }

public virtual ICollection<Appointment> Appointments { get; set; }

public virtual Company Company { get; set; }

}

public class Appointment

{

public int AppointmentId { get; set; }

public System.DateTime AppointmentDate { get; set; }

public string Patient { get; set; }

public int DoctorId { get; set; }

public virtual Doctor Doctor { get; set; }

}

接下来,在Listing 13-7,我们创建用CodeFirst方式时访问EF的途径,DbContext对象

Listing 13-7. DbContext Object

public class Recipe3Context : DbContext

{

public Recipe3Context()

: base("Recipe3ConnectionString")

{

// Disable Entity Framework Model Compatibility

Database.SetInitializer<Recipe3Context>(null);

}

protected override void OnModelCreating(DbModelBuilder modelBuilder)

{

modelBuilder.Entity<Appointment>().ToTable("Chapter13.Appointment");

modelBuilder.Entity<Company>().ToTable("Chapter13.Company");

modelBuilder.Entity<Doctor>().ToTable("Chapter13.Doctor");

}

public DbSet<Appointment> Appointments { get; set; }

public DbSet<Company> Companies { get; set; }

public DbSet<Doctor> Doctors { get; set; }

}

接下来我们在项目中添加App.Config,并把下列Listing 13-8的代码加入到ConnectionString节下

Listing 13-8. Connection String

<connectionStrings>

<add name="Recipe3ConnectionString"

connectionString="Data Source=.;

Initial Catalog=EFRecipes;

Integrated Security=True;

MultipleActiveResultSets=True"

providerName="System.Data.SqlClient" />

</connectionStrings>

为了获取Doctors和Companies并使他们不添加到上下文对象中,我们把AsNoTracking方法链接到获取实体的查询中,如Listing 13-9 那样.

Listing 13-9. Doing a Simple Query Using the AsNoTracking Method

using (var context = new Recipe3Context())

{

var company = new Company { Name = "Paola Heart Center" };

var doc1 = new Doctor { Name = "Jill Mathers", Company = company };

var doc2 = new Doctor { Name = "Robert Stevens", Company = company };

var app1 = new Appointment

{

AppointmentDate = DateTime.Parse("3/18/2010"),

Patient = "Karen Rodgers",

Doctor = doc1

};

var app2 = new Appointment

{

AppointmentDate = DateTime.Parse("3/20/2010"),

Patient = "Steven Cook",

Doctor = doc2

};

context.Doctors.Add(doc1);

context.Doctors.Add(doc2);

context.Appointments.Add(app1);

context.Appointments.Add(app2);

context.Companies.Add(company);

context.SaveChanges();

}

using (var context = new Recipe3Context())

{

Console.WriteLine("Entities tracked in context for Doctors...");

// 用AsNoTracking() 方法执行查询

context.Doctors.Include("Company").AsNoTracking().ToList();

Console.WriteLine("Number of entities loaded into context with AsNoTracking: {0}",

context.ChangeTracker.Entries().Count());//输出:0

// 不用AsNoTracking() 方法执行查询

context.Doctors.Include("Company").ToList();

Console.WriteLine("Number of entities loaded into context without AsNoTracking: {0}",

context.ChangeTracker.Entries().Count());//输出:3

}

输出结果如下:

Entities tracked in context for Doctors...

Number of entities loaded into context with AsNoTracking: 0

Number of entities loaded into context without AsNoTracking: 3

它是如何工作的

当我们把AsNoTracking方法链接上你的查询,从查询返回的结果不会被上下文跟踪.在我们的例子中,我们显示地Include了Doctor的Comanpies.

默认情况下,你的查询结果会被上下文跟踪,这使得更新和删除更容易,但是代价是付出更多的内存和CPU负载.为应用程序串连更多的对象,比如在电子商务网站上浏览产品,使用AsNoTracking选项,能节省更多的资源,使应用程序性能更高.

没有缓存一个查询结果,你每次都得为查询实例化.通常地,使修改跟踪可用,EF将不需要为一个上下文中已经存在的实例再实例化.

当你包含AsNoTracking选项(如我们的Listing 13-9),它只影响当前查询的实体.它不会影响后面的不包含AsNoTracking选项的查询,如Listing 13-9 演示的

Entity Framework 6 Recipes 2nd Edition(13-3)译 -> 为一个只读的访问获取实体的更多相关文章

  1. Entity Framework 6 Recipes 2nd Edition 译 -> 目录 -持续更新

    因为看了<Entity Framework 6 Recipes 2nd Edition>这本书前面8章的翻译,感谢china_fucan. 从第九章开始,我是边看边译的,没有通读,加之英语 ...

  2. Entity Framework 6 Recipes 2nd Edition(9-1)译->用Web Api更新单独分离的实体

    第九章 在N层结构的应用程序中使用EF 不是所有的应用都能完全地写入到一个单个的过程中(就是驻留在一个单一的物理层中),实际上,在当今不断发展的网络世界,大量的应用程序的结构包含经典的表现层,应用程, ...

  3. Entity Framework 6 Recipes 2nd Edition(9-3)译->找出Web API中发生了什么变化

    9-3. 找出Web API中发生了什么变化 问题 想通过基于REST的Web API服务对数据库进行插入,删除和修改对象图,而不必为每个实体类编写单独的更新方法. 此外, 用EF6的Code Fri ...

  4. Entity Framework 6 Recipes 2nd Edition(9-4)译->Web API 的客户端实现修改跟踪

    9-4. Web API 的客户端实现修改跟踪 问题 我们想通过客户端更新实体类,调用基于REST的Web API 服务实现把一个对象图的插入.删除和修改等数据库操作.此外, 我们想通过EF6的Cod ...

  5. Entity Framework 6 Recipes 2nd Edition(13-2)译 -> 用实体键获取一个单独的实体

    问题 不管你用DBFirst,ModelFirst或是CodeFirst的方式,你想用实体键获取一个单独的实体.在本例中,我们用CodeFirst的方式. 解决方案 假设你有一个模型表示一个Paint ...

  6. Entity Framework 6 Recipes 2nd Edition(13-4)译 -> 有效地创建一个搜索查询

    问题 你想用LINQ写一个搜索查询,能被转换成更有效率的SQL.另外,你想用EF的CodeFirst方式实现. 解决方案 假设你有如下Figure 13-6所示的模型 Figure 13-6. A s ...

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

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

  8. Entity Framework 6 Recipes 2nd Edition(13-8)译 -> 把昂贵的属性移到其它实体

    问题 你想把一个昂贵的属性移到另一个实体,这样你就可以延迟加载当前这个实体.对于一个加载昂贵的而且很少用到的属性尤其有用. 解决方案 模型和上一节(Recipes 13-7)的一致,如Figure13 ...

  9. Entity Framework 6 Recipes 2nd Edition(13-9)译 -> 避免Include

    问题 你想不用Include()方法,立即加载一下相关的集合,并想通过EF的CodeFirst方式实现. 解决方案 假设你有一个如Figure 13-14所示的模型: Figure 13-14. A ...

随机推荐

  1. 【造轮子】打造一个简单的万能Excel读写工具

    大家工作或者平时是不是经常遇到要读写一些简单格式的Excel? shit!~很蛋疼,因为之前吹牛,就搞了个这东西,还算是挺实用,和大家分享下. 厌烦了每次搞简单类型的Excel读写?不怕~来,喜欢流式 ...

  2. AndroidTips-052:.aar文件依赖

    aar aar 文件是android 类库项目的输出文件,其中可以包含普通的.class,清单,以及android项目特有的资源文件. 使用方式 将.aar文件放在在自己项目的libs目录下 在gra ...

  3. vmware里面的名词 vSphere、vCenter Server、ESXI、vSphere Client

    vmware里面的名词 vSphere.vCenter Server.ESXI.vSphere Client vSphere.vCenter Server.ESXI.vSphere Client VS ...

  4. Android 判断一个 View 是否可见 getLocalVisibleRect(rect) 与 getGlobalVisibleRect(rect)

    Android 判断一个 View 是否可见 getLocalVisibleRect(rect) 与 getGlobalVisibleRect(rect) [TOC] 这两个方法的区别 View.ge ...

  5. Convert BSpline Curve to Arc Spline in OpenCASCADE

    Convert BSpline Curve to Arc Spline in OpenCASCADE eryar@163.com Abstract. The paper based on OpenCA ...

  6. CRL快速开发框架系列教程十(导出对象结构)

    本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...

  7. Mac OS 使用 Vagrant 管理虚拟机(VirtualBox)

    Vagrant(官网.github)是一款构建虚拟开发环境的工具,支持 Window,Linux,Mac OS,Vagrant 中的 Boxes 概念类似于 Docker(实质是不同的),你可以把它看 ...

  8. css3线条围绕跑马+jquery打字机效果

    原文地址:css3线条围绕跑马+jquery打字机效果 有图有真相,今天偶然看到了一种效果,仔细看了下,发现它是用css的clip+css3的动画实现的,简直叼.于是自己拿来了前一阵子写的打字机效果, ...

  9. System.Guid ToString五中格式

    参考:https://msdn.microsoft.com/en-us/library/97af8hh4.aspx 测试代码: using System; using System.Collectio ...

  10. Django admin定制化,User字段扩展[原创]

    前言 参考上篇博文,我们利用了OneToOneField的方式使用了django自带的user,http://www.cnblogs.com/caseast/p/5909248.html , 但这么用 ...