EF中的贪婪加载和延迟加载(懒加载)
在上一章中,我们使用了Linq对Entity Framework进行了一个查询,但是通过学习我们却发现了懒加载给我来的性能上的开销是很到的,尤其是在循环中,如果数据量不是很多的情况下还可以接受,如果数据量一旦大气来,那么这个效率则是影响非常大的。那该怎么办呢?其实在Entity Framwork中,除了提供了懒加载技术还提供了一个“贪婪加载”。那么什么是贪婪加载呢?从名字上看,就是非常的粗鲁的,一次性的吧相关的数据全部查询出来,虽然在性能上说还是有点影响的,但是比起在循环中使用懒加载要强了不少了啊。
下面呢,老魏先不说懒加载的知识,把上一张遗留的一个查询给家说一下,顺便以这个例子和贪婪加载做一下对比。
Demo3:查询班级的信息,还要得到此班级中的学生。
SQL:
select a.* ,b.* from clazz as a left join student as b on a.CId = b.CId
Linq:
DAL.SchoolContext context = new DAL.SchoolContext();
var query = from clazz in context.Clazz
select clazz;
foreach (var clazz in query)
{
Console.WriteLine(clazz.CName);
if (clazz.Students != null && clazz.Students.Count > 0)
{
foreach(var student in 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次)
exec sp_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;
var query = from clazz in context.Clazz.Include("Students")
select clazz;
foreach (var clazz in query)
{
Console.WriteLine(clazz.CName);
if (clazz.Students != null && clazz.Students.Count > 0)
{
foreach(var student in 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],
CASE WHEN ([Extent2].[SId] IS NULL) THEN CAST(NULL AS int) ELSE 1 END AS [C1]
FROM [dbo].[Clazz] AS [Extent1]
LEFT OUTER JOIN [dbo].[Student] AS [Extent2] ON [Extent1].[CId] = [Extent2].[CId]
) AS [Project1]
ORDER BY [Project1].[CId] ASC, [Project1].[C1] ASC
SQL Server状态:

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

无非就是在Linq中加入了一个Include()方法。这个方法就是用来开启贪婪加载的主要方法。意思是说在加载查询对象的时候。把查询对象的关联数据也查询出来。其实这里面是有陷阱的。当然这陷阱值得是Include的参数。顾名思义,和clazz管理的是student对象,那么在参数就是”Student”。其实不然,因为根据Include参数的含义是说“路径”。那么这个路径是什么呢?其实就是”导航属性的名字“。在clazz中有一个导航属性是Students,则在Include中也要使用这个名字。
当然了,如果大家想的到的话,那么和Student关联的对象能够查询出来呢?答案是肯定的,如果关联属性有多个则使用”.”来连接。比如我们可以把代码改为如下的样子:
var query = from clazz in context.Clazz.Include("Students.Student_Courses.Courses")
select clazz;
那我们在查询clazz对象的也查出来了student,course的信息。其实看到这里大家就知道了贪婪加载虽然没有在循环中那么的消耗性能,但是一次性查询的数据是很多的,还是有影响的,但是没有懒加载那么厉害了。
总结一下,如果要在循环中使用数据,请使用贪婪加载,否则使用懒加载。
EF中的贪婪加载和延迟加载(懒加载)的更多相关文章
- Android中ViewPager+Fragment取消(禁止)预加载延迟加载(懒加载)问题解决方案
转载请注明出处:http://blog.csdn.net/linglongxin24/article/details/53205878本文出自[DylanAndroid的博客] Android中Vie ...
- jQuery延迟加载(懒加载)插件 – jquery.lazyload.js-Web前端(W3Cways.com) - Web前端学习之路
Lazy Load 是一个用 JavaScript 编写的 jQuery 插件. 它可以延迟加载长页面中的图片. 在浏览器可视区域外的图片不会被载入, 直到用户将页面滚动到它们所在的位置. 这与图片预 ...
- vue 中监测滚动条加载数据(懒加载数据)
vue 中监测滚动条加载数据(懒加载数据) 1:钩子函数监听滚动事件: mounted () { this.$nextTick(function () { window.addEventListene ...
- vue路由懒加载及组件懒加载
一.为什么要使用路由懒加载 为给客户更好的客户体验,首屏组件加载速度更快一些,解决白屏问题. 二.定义 懒加载简单来说就是延迟加载或按需加载,即在需要的时候的时候进行加载. 三.使用 常用的懒加载方式 ...
- jQuery延迟加载(懒加载)插件 – jquery.lazyload.js
引入:<script type="text/javascript" src="${base}/resources/shop/js/jquery.lazyload.j ...
- OC中重写set和get方法、懒加载
在写OC程序的时候,在很多时候我们会用到重写set或者get方法,重写这两个方法大多是用于刷新数据,比如懒加载. 意思就是说当你去调用set或者get方法时,系统会去调用重写的get或者set方法,这 ...
- 图片利用 new Image()预加载原理 和懒加载的实现原理
二:预加载和懒加载的区别 预加载与懒加载,我们经常经常用到,这些技术不仅仅限于图片加载,我们今天讨论的是图片加载: 图片预加载:顾名思义,图片预加载就是在网页全部加载之前,提前加载图片.当用户需要查看 ...
- 滚动加载图片(懒加载)实现原理(这是旧实现,仅做为获取元素宽高api的参考)
https://www.cnblogs.com/flyromance/p/5042187.html 本文主要通过以下几方面来说明懒加载技术的原理,个人前端小菜,有错误请多多指出 一.什么是图片滚动加载 ...
- el-select下拉加载(实现懒加载)
情况:项目出现了下拉数据量过大,出现页面卡死问题,反馈到我这:当时实现思路1.使用render函数去渲染下拉框 试了发现卡死情况依然存在,所以尝试方法2 2.使用原生js去添加下拉框的<opti ...
随机推荐
- 乐在其中设计模式(C#) - 建造者模式(Builder Pattern)
原文:乐在其中设计模式(C#) - 建造者模式(Builder Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 建造者模式(Builder Pattern) 作者:webabc ...
- MongoDB学习笔记-命令
连接数据库: mongodb://账号:密码@IP/库名 更多方式参考:http://www.runoob.com/mongodb/mongodb-connections.html 命令整理: 名称 ...
- 我收集的sonar参考资料
sonarQube代码质量管理工具环境筹建笔记 http://www.myexception.cn/open-source/1307345.html 配置sonar.jenkins进行持续审查 htt ...
- 采用objdump调试驱动程序
最近的一个推断调整nand是好是坏司机+测试程序,因此,与下面的调整过程.看来他也学到了一点知识.因此,关于备案. 这篇文章主要是讲述调式驱动的一个方法而已. 先来看看測试程序 #include &l ...
- 如何找到w3wp与w3svc的对应关系
在生产环境中,一般会有多个IIS进程在运行,这里面可能是有Web Garden的设置,也可能是有多个application pool在运行.而我们经常在c:\inetpub\logs目录下面,看到很多 ...
- EasyUI禁用控制方法常采用
EasyUI禁用控制方法常采用: 1.validatebox使用可以使用:前两个适用于个人validatebox; 第三适用于整个form内箱; <1>.$("#id& ...
- implements KeyListener但关键监听器监听少
今天写的游戏.主要听众,但它并不总是加入了育雏, 我实现了接口,但不听 后来,我发现只是没想到服用口服细致怎么称呼控制panel上面增加了一个addKeyListener(this); 基础不坚固.马 ...
- MySQL存储过程:用户授权量
写这些脚本需求放缓的调查记录到数据库,方便观看. 1. 因为默认mysql.slow_log表使用csv数据引擎,该数据不支持指数,因此,有必要改变MyISAM发动机.和query_time字段索引, ...
- 谈话节目APE系列:如何成为技术达人
作为一个程序猿,总有消退的前辈.或更年轻的同行.牛逼的人总是羡慕. 让我们搞自己痛苦的日子 BUG .头发很快结束了抓,人们扫两.修改一行代码.问题得以克服:例如,他们自己开发的十年,少付 10K , ...
- Hadoop Hive sql 语法详细解释
Hive 是基于Hadoop 构建的一套数据仓库分析系统.它提供了丰富的SQL查询方式来分析存储在Hadoop 分布式文件系统中的数据,能够将结构 化的数据文件映射为一张数据库表,并提供完整的SQL查 ...