Entity Framework学习笔记(五)----Linq查询(2)---贪婪加载
请注明转载地址:http://www.cnblogs.com/arhat
在上一章中,我们使用了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 > )
{
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 > )
{
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的信息。其实看到这里大家就知道了贪婪加载虽然没有在循环中那么的消耗性能,但是一次性查询的数据是很多的,还是有影响的,但是没有懒加载那么厉害了。
总结一下,如果要在循环中使用数据,请使用贪婪加载,否则使用懒加载。本章就到这里了。希望大家能够顶一下!谢谢了。
Entity Framework学习笔记(五)----Linq查询(2)---贪婪加载的更多相关文章
- Entity Framework学习笔记(四)----Linq查询(1)
请注明转载地址:http://www.cnblogs.com/arhat 从本章开始,老魏就介绍一下Entity Framework使用Linq来查询数据,也就是Linq To Entity.其实在E ...
- Entity Framework学习笔记
原文地址:http://www.cnblogs.com/frankofgdc/p/3600090.html Entity Framework学习笔记——错误汇总 之前的小项目做完了,到了总结经验和 ...
- ADO.NET Entity Framework学习笔记(3)ObjectContext
ADO.NET Entity Framework学习笔记(3)ObjectContext对象[转] 说明 ObjectContext提供了管理数据的功能 Context操作数据 AddObject ...
- Entity Framework 学习笔记(2)
上期回顾:Entity Framework 学习笔记(1) Entity Framework最主要的东西,就是自己创建的.继承于DbContext的类: /// <summary> /// ...
- 驱动开发学习笔记. 0.07 Uboot链接地址 加载地址 和 链接脚本地址
驱动开发学习笔记. 0.07 Uboot链接地址 加载地址 和 链接脚本地址 最近重新看了乾龙_Heron的<ARM 上电启动及 Uboot 代码分析>(下简称<代码分析>) ...
- Entity Framework学习笔记(六)----使用Lambda查询Entity Framework(1)
请注明转载地址:http://www.cnblogs.com/arhat 在前几章中,老魏一直使用Linq来查询Entity Framework.但是老魏感觉,如果使用Linq的话,那么Linq的返回 ...
- Entity Framework学习笔记——错误汇总
之前的小项目做完了,到了总结经验和更新学习笔记的时间了.开始正题之前先啰嗦一下,对之前的学习目标进行一个调整:“根据代码生成表”与“生成数据库脚本和变更脚本”合并为“Code First模式日常使用篇 ...
- Entity Framework学习笔记(二)----CRUD(1)
请注明转载地址:http://www.cnblogs.com/arhat 这篇文章老魏和大家分享一下Entity Framework的CRUD操作,在这之前呢,老魏先说一下老魏对EF的一个整体的认识, ...
- Entity Framework 学习笔记
1.自定义数据库链接字符串上下文 public class PetDbContext : DbContext { public PetDbContext() : base("name=Dem ...
随机推荐
- JNA结构体参数传递,Java数组
JNA以结构体数组为参数进行调用: ////// C++ // student 结构体定义 typedef struct { int age; char name[20]; }Student; // ...
- 六.CSS浮动与清除
浮动 把元素从常规文档流中取出.是元素脱离常规文档流 浮动的作用: ①实现文本绕排图片效果 ②让原本在上下方向上堆叠的块级元素可以变成左右并列,从而实现多栏布局 文本绕排图片 首先HTML代码如下 & ...
- SQLServer排序时与读取的记录会影响到结果?
这是在做程序的时候发现的,我用到了一个分页存储过程,在翻看第二页的时候发现结果竟然与第一页有很多重复的内容, 下面开始测试一下吧: 创建表 create table abc ( id int prim ...
- php中sprintf与printf函数用法区别
下面是一个示例:四舍五入保留小数点后两位 代码如下 复制代码 <?php$num1 = 21;echo sprintf("%0.2f",$num1)."<b ...
- 理解C#系列 / C#语言的特性
C#语言的特性 大多数语句都已(;)结尾 用({})定义语句块 单行注释(//),多行注释(/*......*/)智能注释(///) 区分大小写 用namespace名称空间对类进行分类 在C#中的所 ...
- C语言获取系统时间的几种方式[转]
C语言获取系统时间的几种方式 C语言中如何获取时间?精度如何? 1 使用time_t time( time_t * timer ) 精确到秒 2 使用clock_t clock() 得到的是CPU时间 ...
- arcgis js api 本地化配置
配置arcgis library 根目录的init.js的 "baseUrl:",使其指向正确的地址
- JS源码(条件的判定,循环,数组,函数,对象)整理摘录
--- title: JS学习笔记-从条件判断语句到对象创建 date: 2016-04-28 21:31:13 tags: [javascript,front-end] ---JS学习笔记——整理自 ...
- Linux概述
Linux概述 1.计算机资源 硬件资源 软件资源 硬件资源与软件资源之间的桥梁就是操作系统 2.操作系统分类 Windows :个人版用户最多 Mac :土豪机 Linux :主要应用于服务器 Un ...
- 济南学习 Day 3 T1 pm
巧克力棒(chocolate)Time Limit:1000ms Memory Limit:64MB题目描述LYK 找到了一根巧克力棒,但是这根巧克力棒太长了,LYK 无法一口吞进去.具体地,这根巧克 ...