数据加载分为延迟加载和预加载
EF的关联实体加载有三种方式:Lazy Loading,Eager Loading,Explicit Loading,其中Lazy Loading和Explicit Loading都是延迟加载。
(一)Lazy Loading使用的是动态代理,关闭Lazy Loading,可以将LazyLoadingEnabled设为false,如果导航属性没有标记为virtual,Lazy Loading也是不起作用的。
(二)Eager Loading使用Include方法关联预先加载的实体。
(三)Explicit Loading使用Entry方法,对于集合使用Collection,单个实体则使用Reference。
延迟加载:
以上一章的产品和发票为例:
//延迟加载
context.Configuration.ProxyCreationEnabled = true;
context.Configuration.LazyLoadingEnabled = true;
//此时不会加载SysInvoice属性关联的对象
var productGet = context.Set<Product>().First(r => r.Id == );
//直到用到SysInvoice时,才会新起一个查询获取Invoice
var date = productGet.SysInvoice.CreateDate;
作为默认配置的延迟加载,需要满足以下几个条件:
context.Configuration.ProxyCreationEnabled = true;
context.Configuration.LazyLoadingEnabled = true;
导航属性被标记为virtual
这三个条见缺一不可。(把导航属性virtual去掉可以禁用单个实体的延迟加载。)
如果不满足条件,延迟加载则不会启用,这时候我们必须使用手动加载的方式来获取关联数据。
手动加载就是通过DbReferenceEntry的Load方法来实现。我们把设置context.Configuration.LazyLoadingEnabled = false;(全局禁用延迟加载)以便在没有延迟加载的环境进行测试。
//关闭延迟加载,手动加载(并且去掉SysInvoice导航属性virtual)
context.Configuration.ProxyCreationEnabled = false;
context.Configuration.LazyLoadingEnabled = false; //此时不会加载SysInvoice属性关联的对象
var productGet2 = context.Set<Product>().First(r => r.Id == );
默认情况下我们很少会直接用到Load方法,一般ToList或First这样的方法就帮我们完成加载数据操作了。
预加载:
预加载就是使用Include方法并传入需要同时获取的关联属性。我们也可以使用字符串传入属性的名称
//预加载
var d = context.Set<Product>().Include(c=>c.SysInvoice).ToList();
//可以使用字符串传入属性的名称,但是这样会很容易出错,使用lambda能更好的避免错误
var product = context.Set<Product>().Include("SysInvoice").FirstOrDefault();
预加载也支持同时加载二级属性,如果给发票添加一个开票人信息
//预加载也支持同时加载二级属性,比如我们给Invoice增加一个开票人属性,这是一个Employee对象
var product2 = context.Set<Product>().Include(p => p.SysInvoice.Drawer).FirstOrDefault();
var product3 = context.Set<Product>().Include("SysInvoice.Drawer").FirstOrDefault();
显式加载:
使用Entry方法,对于集合使用Collection,单个实体则使用Reference。
//手动加载sysinvoice
context.entry(productget2).reference(p => p.sysinvoice).load();
var date2 = productget2.sysinvoice.createdate; //手动加载集合属性也类似,就是把reference方法换成collection方法
//此时不会加载photos属性关联的对象
var productget = context.set<product>().first(r => r.id == );
//手动加载photos集合
context.entry(productget).collection(p => p.photos).load();
var count = productget.photos.count;