EF中的贪婪加载和延迟加载(懒加载)

在上一章中,我们使用了Linq对Entity Framework进行了一个查询,但是通过学习我们却发现了懒加载给我来的性能上的开销是很到的,尤其是在循环中,如果数据量不是很多的情况下还可以接受,如果数据量一旦大气来,那么这个效率则是影响非常大的。那该怎么办呢?其实在Entity Framwork中,除了提供了懒加载技术还提供了一个“贪婪加载”。那么什么是贪婪加载呢?从名字上看,就是非常的粗鲁的,一次性的吧相关的数据全部查询出来,虽然在性能上说还是有点影响的,但是比起在循环中使用懒加载要强了不少了啊。

下面呢,老魏先不说懒加载的知识,把上一张遗留的一个查询给家说一下,顺便以这个例子和贪婪加载做一下对比。

Demo3:查询班级的信息,还要得到此班级中的学生。

SQL:

selecta.*,b.*fromclazzasaleftjoinstudentas bona.CId=b.CId

Linq:

DAL.SchoolContext context =new DAL.SchoolContext();varquery =fromclazzin context.Clazzselect clazz;foreach(varclazzin query)

{

      Console.WriteLine(clazz.CName);if(clazz.Students !=null&& clazz.Students.Count >0)

  {        foreach(varstudentin clazz.Students)

        {

          Console.WriteLine("---该班的学生:"+ student.SName);

      }

  }

}

翻译SQL:在执行中翻译的SQL

这里老魏截图了,就是因为在循环中使用懒加载而产生了n多个查询语句,但是

总体上两个SQL语句:

1,查询出clazz信息的SQL

SELECT[Extent1].[CId]AS[CId],

    [Extent1].[CName]AS[CName]FROM[dbo].[Clazz]AS[Extent1]

2,根据懒加载而产生的SQL语句(被重复了N次)

execsp_executesql N'SELECT

    [Extent1].[SId] AS [SId],

    [Extent1].[SName] AS [SName],

    [Extent1].[SAge] AS [SAge],

    [Extent1].[SMail] AS [SMail],

    [Extent1].[CId] AS [CId]

    FROM [dbo].[Student] AS [Extent1]

    WHERE [Extent1].[CId] = @EntityKeyValue1',N'@EntityKeyValue1 int',@EntityKeyValue1=2

虽然我们通过懒加载可以达到我们想要的效果,但是在效率上是无法忍受的尤其是在数据多的情况下。

那么Entity Framework也想到了这问题,因为所有的ORM框架都有懒加载,使用起来的确的方便,但是也是在改用的时候用,尤其在循环中就更不应该使用了。所以这里呢,老魏有个建议,就是在循环中千万不要使用懒加载。既然在循环中不能使用懒加载那么该怎么办呢?这就要利用Entity Framework给我们提供的贪婪加载。下面看以下代码,然后老魏在来解释一下。把上面的代码改为如下的代码:

DAL.SchoolContext context =new DAL.SchoolContext();//取消懒加载目的是为了做实验看能够一次加载完数据context.Configuration.LazyLoadingEnabled =false;varquery =fromclazzincontext.Clazz.Include("Students")select clazz;foreach(varclazzin query)

{

          Console.WriteLine(clazz.CName);          if(clazz.Students !=null&& clazz.Students.Count >0)

          {              foreach(varstudentin clazz.Students)

            {

                Console.WriteLine("---该班的学生:"+ student.SName);

              }

        }

}

运行一下,同时监控一下SQL Server的状态。首先是翻译的SQL:

SELECT[Project1].[CId]AS[CId],

    [Project1].[CName]AS[CName],

    [Project1].[C1]AS[C1],

    [Project1].[SId]AS[SId],

    [Project1].[SName]AS[SName],

    [Project1].[SAge]AS[SAge],

    [Project1].[SMail]AS[SMail],

    [Project1].[CId1]AS[CId1]FROM(SELECT[Extent1].[CId]AS[CId],

        [Extent1].[CName]AS[CName],

        [Extent2].[SId]AS[SId],

        [Extent2].[SName]AS[SName],

        [Extent2].[SAge]AS[SAge],

        [Extent2].[SMail]AS[SMail],

        [Extent2].[CId]AS[CId1],

        CASEWHEN([Extent2].[SId]ISNULL)THENCAST(NULLASint)ELSE1ENDAS[C1]FROM[dbo].[Clazz]AS[Extent1]LEFTOUTERJOIN[dbo].[Student]AS[Extent2]ON[Extent1].[CId]=[Extent2].[CId]    )  AS[Project1]ORDERBY[Project1].[CId]ASC,[Project1].[C1]ASC

SQL Server状态:

发现在执行的过程中,循环语句并没有的发出额外的指令,只是用来上面翻译的SQL。但是结果却是一样的。

无非就是在Linq中加入了一个Include()方法。这个方法就是用来开启贪婪加载的主要方法。意思是说在加载查询对象的时候。把查询对象的关联数据也查询出来。其实这里面是有陷阱的。当然这陷阱值得是Include的参数。顾名思义,和clazz管理的是student对象,那么在参数就是”Student”。其实不然,因为根据Include参数的含义是说“路径”。那么这个路径是什么呢?其实就是”导航属性的名字“。在clazz中有一个导航属性是Students,则在Include中也要使用这个名字。

当然了,如果大家想的到的话,那么和Student关联的对象能够查询出来呢?答案是肯定的,如果关联属性有多个则使用”.”来连接。比如我们可以把代码改为如下的样子:

varquery =fromclazzincontext.Clazz.Include("Students.Student_Courses.Courses")                selectclazz;

那我们在查询clazz对象的也查出来了student,course的信息。其实看到这里大家就知道了贪婪加载虽然没有在循环中那么的消耗性能,但是一次性查询的数据是很多的,还是有影响的,但是没有懒加载那么厉害了。

总结一下,如果要在循环中使用数据,请使用贪婪加载,否则使用懒加载。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,734评论 6 505
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,931评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,133评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,532评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,585评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,462评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,262评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,153评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,587评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,792评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,919评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,635评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,237评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,855评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,983评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,048评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,864评论 2 354

推荐阅读更多精彩内容

  • 希望对大家面试有帮助,很经典的几道题,大家有兴趣的话,欢迎大家加群讨论,QQ群号:295383988 问题及描述:...
    hc爱编程阅读 1,891评论 0 12
  • Student(Sid,Sname,Sage,Ssex) 学生表Course(Cid,Cname,Tid) 课程表...
    小炉炉阅读 1,451评论 0 0
  • 一、多表查询 --编写多表查询语句的一般过程 --(1)、分析句子要涉及到哪些表 --(2)、对应的表中要查询哪些...
    __71db阅读 1,830评论 0 6
  • 说明:以下五十个语句都按照测试数据进行过测试,最好每次只单独运行一个语句。 问题及描述: --1.学生表 Stud...
    lijun_m阅读 1,304评论 0 1
  • 最常见的三种模式 1.资本增值 常见形式有保险、企业理财、股票证券等。 2.电子交易 如微信支付、支付宝支付,产生...
    Jim花阅读 103评论 0 0