EF的关联实体加载有三种方式

6 篇文章 0 订阅

EF的关联实体加载有三种方式:Lazy Loading,Eager Loading,Explicit Loading,其中Lazy Loading和Explicit Loading都是延迟加载。

(一)延迟加载(默认):Lazy Loading使用的是动态代理,默认情况下,如果POCO类满足以下两个条件,EF就使用Lazy Loading:

POCO类是Public且不为Sealed。
导航属性标记为Virtual。
  关闭Lazy Loading,可以将LazyLoadingEnabled设为false,如果导航属性没有标记为virtual,Lazy Loading也是不起作用的。

(二)贪婪加载:不设置导航属性为virtual,并且对导航属性使用Include,Eager Loading使用Include方法关联预先加载的实体。

(三)显示加载:不设置导航属性为virtual,并且对导航属性使用Reference(单个对象).Load()或Collection(对象集).Load()。Explicit Loading使用Entry方法,对于集合使用Collection,单个实体则使用Reference。

public class Programm
    {
        public static void Main()
        {
            TestDbContext db = new TestDbContext();

            //延迟加载
            var res1 = db.Reservations;
            foreach (var item in res1)
            {
                //do something
            }

            //贪婪加载
            //有多个Include会全部完成加载,生成的sql命令是一条完成的
            var res2 = db.Reservations.Include(r => r.Details);

            //显示加载
            var res3 = db.Reservations.ToList();
            foreach (var item in res3)
            {
                var temp = db.Entry(item);
                temp.Collection(i => i.Details).Load();
                temp.Reference(i => i.Customer).Load();
            }
        }
    }

    public class Reservation
    {
        public int ReservationId { get; set; }
        public string ClientName { get; set; }
        public string Location { get; set; }
        public List<ReservationDetails> Details { get; set; }
        public Customer Customer { get; set; }
    }

    public class ReservationDetails
    {
        public int Id { get; set; }
        public DateTime Time { get; set; }
    }

    public class Customer
    {
        public int Id { get; set; }
        public string Name { get; set; }
    }

    public class TestDbContext : DbContext
    {
        public DbSet<Reservation> Reservations { get; set; }
    }

此外,可以在EF的配置文件中设备加载方式 ,ProxyCreationEnabled,LazyLoadingEnabled 我们会使用这两个设置来决定是否加载导航属性。默认情况这两个值都是true的,也就是说会以延迟加载的方式加载导航属性,也就是当我们访问导航属性的时候才会去查数据库获取导航属性的数据。
db.Configuration.ProxyCreationEnabled = true ;
db.Configuration.LazyLoadingEnabled = true ;

想禁用加载实体的导航属性,可以这样设置:
db.Configuration.ProxyCreationEnabled = true ;
db.Configuration.LazyLoadingEnabled = false;

或者直接:
db.Configuration.ProxyCreationEnabled = false;

需要注意的是,当ProxyCreationEnabled =false的时候,LazyLoadingEnabled 是不起作用的。

 public class MYDB:DbContext
    {
    	...
 		public MYDB(DbConnection existingConnection, bool contextOwnsConnection) :
            base(existingConnection, contextOwnsConnection)
        {
            ConfigurationFunc();
        }

        private void ConfigurationFunc()
        {
            Configuration.LazyLoadingEnabled = true;
            Configuration.ProxyCreationEnabled = true;
        }
        ...
  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值