Entity Framework关联查询以及数据加载(延迟加载,预加载)

时间:2023-08-09 14:24:56

数据加载分为延迟加载和预加载

EF的关联实体加载有三种方式:Lazy Loading,Eager Loading,Explicit Loading,其中Lazy LoadingExplicit 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;

作为默认配置的延迟加载,需要满足以下几个条件:

  1. context.Configuration.ProxyCreationEnabled = true;

  2. context.Configuration.LazyLoadingEnabled = true;

  3. 导航属性被标记为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;