《Entity Framework 6 Recipes》中文翻译系列 (29) ------ 第五章 加载实体和导航属性之过滤预先加载的实体集合和修改外键关联
翻译的初衷以及为什么选择《Entity Framework 6 Recipes》来学习,请看本系列开篇
5-13 过滤预先加载的实体集合
问题
你想过滤预先加载的实体集合,另外,你想使用Code-First来管理数据访问
解决方案
实体框架不支持直接使用Include()时过滤关联实体集合,但我们可以通过创建一个匿名类型来完成同样的事情,匿名类型包含实体和要过滤的关联实体集合。
假设你有如图5-28所示的概念模型

图5-28 一个包含movies(电影)和它的categories(目录)的模型
在Visual Studio中添加一个名为Recipe13的控制台应用,并确保引用了实体框架6的库,NuGet可以很好的完成这个任务。在Reference目录上右键,并选择 Manage NeGet Packages(管理NeGet包),在Online页,定位并安装实体框架6的包。这样操作后,NeGet将下载,安装和配置实体框架6的库到你的项目中。
接下来我们创建两个实体对象:Moviet和Category,复制代码清单5-33中的属性到这三个类中。
代码清单5-33. 实体类
public class Category
{
public Category()
{
Movies = new HashSet<Movie>();
} public int CategoryId { get; set; }
public string Name { get; set; }
public string ReleaseType { get; set; } public virtual ICollection<Movie> Movies { get; set; }
} public class Movie
{
public int MovieId { get; set; }
public string Name { get; set; }
public string Rating { get; set; }
public int CategoryId { get; set; } public virtual Category Category { get; set; }
}
接下来,创建一个名为Recipe13Context的类,并将代码清单5-34中的代码添加到其中,并确保其派生到DbContext类。
代码清单5-34. 上下文
public class Recipe13Context : DbContext
{
public Recipe13Context()
: base("Recipe13ConnectionString")
{
//禁用实体框架的模型兼容
Database.SetInitializer<Recipe13Context>(null);
} public DbSet<Category> Categories { get; set; }
public DbSet<Movie> Movies { get; set; } protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Entity<Category>().ToTable("Chapter5.Category");
modelBuilder.Entity<Movie>().ToTable("Chapter5.Movie");
}
}
接下来添加App.Config文件到项目中,并使用代码清单5-35中的代码添加到文件的ConnectionStrings小节下。
代码清单5-35. 连接字符串
<connectionStrings>
<add name="Recipe13ConnectionString"
connectionString="Data Source=.;
Initial Catalog=EFRecipes;
Integrated Security=True;
MultipleActiveResultSets=True"
providerName="System.Data.SqlClient" />
</connectionStrings>
为了过滤categoryes和与之关联的movies,请按代码清单5-36的方式进行。
代码清单5-36.过滤预先加载的实体集合
using (var context = new Recipe13Context())
{
var cat1 = new Category {Name = "Science Fiction", ReleaseType = "DVD"};
var cat2 = new Category {Name = "Thriller", ReleaseType = "Blu-Ray"};
var movie1 = new Movie {Name = "Return to the Moon", Category = cat1, Rating = "PG-13"};
var movie2 = new Movie {Name = "Street Smarts", Category = cat2, Rating = "PG-13"};
var movie3 = new Movie {Name = "Alien Revenge", Category = cat1, Rating = "R"};
var movie4 = new Movie {Name = "Saturday Nights", Category = cat1, Rating = "PG-13"};
context.Categories.Add(cat1);
context.Categories.Add(cat2);
context.Movies.Add(movie1);
context.Movies.Add(movie2);
context.Movies.Add(movie3);
context.Movies.Add(movie4);
context.SaveChanges();
} using (var context = new Recipe13Context())
{
// 通过ReleaseType和Rating过虑
// 创建匿名类型集合
var cats = from c in context.Categories
where c.ReleaseType == "DVD"
select new
{
category = c,
movies = c.Movies.Where(m => m.Rating == "PG-13")
}; Console.WriteLine("PG-13 Movies Released on DVD");
Console.WriteLine("============================");
foreach (var cat in cats)
{
var category = cat.category;
Console.WriteLine("Category: {0}", category.Name);
foreach (var movie in cat.movies)
{
Console.WriteLine("\tMovie: {0}", movie.Name);
}
}
} Console.WriteLine("Press <enter> to continue...");
Console.ReadLine();
代码清单5-36输出如下:
PG- Movies Released on DVD
============================
Category: Science Fiction
Movie: Return to the Moon
Movie: Saturday Nights
原理
代码清单5-36,先创建并初始化categoryes和movies。为了保持简短,我们只创建了两个目录和4部电影。
在查询中,我们创建一个匿名类型的集合,它包含category实例,和在目录中过滤后的movies实例。该查询也过滤了目录集合,只获取发行类型为“DVD"的目录。在示例中,只有一个目录的发行类型为“DVD"。这里我们依赖关系跨度(relationship span)将movies附加到categories。
这个方法凭借匿名对象帮助我们绕开了预先加载中的限制,预先加载不允许我们过滤预先加载的实体集合。注意,正如前面小节中的示例演示的那样,当我们显式加载时,我们能过虑一个预先加载的实体集合。记住,匿名类型对象的生命周期范围只在创建它的方法中,方法不能返回匿名类型。如果我们目标是返回一个应用后面要处理的实体集,那么我们需要创建一个确切的类型来存放数据,然后将它返回。在我们的示例中,这个确切的类应该是一个简单的类,它只创建两个属性:Category 和一个Movies集合。
5-14 修改外键关联
问题
你想修改外键关联。
解决方案
实体框架提供了两种方式来修改外键关联,你可以将一个关联的实体添加到导航属性集合或者赋值给导航属性。你还可以将关联实体的键值设置成外键值。
假设你有如图5-29所示的模型。

图5-29 一个包含client和Invoice的模型
按下面的步骤,使用两种不同的方式,修改client实体和nvoice实体间的外键关联:
1、右键你的项目,选择Add(添加) ➤ADO.NET Entity Data Model(ADO.NET 实体数据模型)。导入Client和Invoice表。确保勾上选项 Include foreign key columns in the model(包含外键列到模型),它默认是勾选上的。如图5-30所示。这样会从数据库中导入外键关联,它不是多对多关系。

图5-30 勾上选项 Include foreign key columns in the model(包含外键列到模型)将会为数据中的关系创建外键关联。数据中的关系不是多对多关系。
2、使用代码清单5-37演示修改外键关联的方法。
代码清单5-37. 演示修改外键关联
using (var context = new EFRecipesEntities())
{
var client1 = new Client {Name = "Karen Standfield", ClientId = };
var invoice1 = new Invoice {InvoiceDate = DateTime.Parse("4/1/10"), Amount = 29.95M};
var invoice2 = new Invoice {InvoiceDate = DateTime.Parse("4/2/10"), Amount = 49.95M};
var invoice3 = new Invoice {InvoiceDate = DateTime.Parse("4/3/10"), Amount = 102.95M};
var invoice4 = new Invoice {InvoiceDate = DateTime.Parse("4/4/10"), Amount = 45.99M}; // 添加一个invoice到client的导航属性集合Invoices
client1.Invoices.Add(invoice1); //直接分配一个外键值
invoice2.ClientId = ; //使用一个“假”实体附加一存在的行
context.Database.ExecuteSqlCommand("insert into chapter5.client values (2, 'Phil Marlowe')");
var client2 = new Client {ClientId = };
context.Clients.Attach(client2);
invoice3.Client = client2; // 使用Client引用
invoice4.Client = client1; //保存更改
context.Clients.Add(client1);
context.Invoices.Add(invoice2);
context.Invoices.Add(invoice3);
context.Invoices.Add(invoice4);
context.SaveChanges();
} using (var context = new EFRecipesEntities())
{
foreach (var client in context.Clients)
{
Console.WriteLine("Client: {0}", client.Name);
foreach (var invoice in client.Invoices)
{
Console.WriteLine("\t{0} for {1}", invoice.InvoiceDate.ToShortDateString(),
invoice.Amount.ToString("C"));
}
}
} Console.WriteLine("Press <enter> to continue...");
Console.ReadLine();
代码清单5-37的输出如下:
Client: Karen Standfield
// for $29.95
// for $45.99
// for $49.95
Client: Phil Marlowe
// for $102.95
原理
实体框架支持独立关联和外键关联。对于独立关联,由关联间的实体自行跟踪,修改关联的唯一方式是通过修改对象引用。
对于外键关联,你可以通过修改对象引用,或是直接修改外键属性值来修改关联。外键关联不能用作多对多关系。
注意 记住,外键关联简单易用,它是默认方法,也是实体框架开发团队推荐的方法。除非你有具体的业务要求使用独立关联。否则请使用外键关联。
表5-1展示了独立关联和外键关联之间的主要区别。
表5-1. 独立关联和外键关联之间的区别

至此第5章结束,下篇我们将进入第六章,感谢你的阅读。觉得有帮助的话,点击下边的推荐以示支持。谢谢!
实体框架交流QQ群: 458326058,欢迎有兴趣的朋友加入一起交流
谢谢大家的持续关注,我的博客地址:http://www.cnblogs.com/VolcanoCloud/
《Entity Framework 6 Recipes》中文翻译系列 (29) ------ 第五章 加载实体和导航属性之过滤预先加载的实体集合和修改外键关联的更多相关文章
- 《Entity Framework 6 Recipes》中文翻译系列 (28) ------ 第五章 加载实体和导航属性之测试实体是否加载与显式加载关联实体
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-11 测试实体引用或实体集合是否加载 问题 你想测试关联实体或实体集合是否已经 ...
- 《Entity Framework 6 Recipes》中文翻译系列 (15) -----第三章 查询之与列表值比较和过滤关联实体
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 3-8与列表值比较 问题 你想查询一个实体,条件是给定的列表中包含指定属性的值. 解 ...
- 《Entity Framework 6 Recipes》中文翻译系列 (22) -----第五章 加载实体和导航属性之延迟加载
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 第五章 加载实体和导航属性 实体框架提供了非常棒的建模环境,它允许开发人员可视化地使 ...
- 《Entity Framework 6 Recipes》中文翻译系列 (23) -----第五章 加载实体和导航属性之预先加载与Find()方法
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-2 预先加载关联实体 问题 你想在一次数据交互中加载一个实体和与它相关联实体. ...
- 《Entity Framework 6 Recipes》中文翻译系列 (24) ------ 第五章 加载实体和导航属性之查询内存对象
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-4 查询内存对象 问题 你想使用模型中的实体对象,如果他们已经加载到上下文中, ...
- 《Entity Framework 6 Recipes》中文翻译系列 (25) ------ 第五章 加载实体和导航属性之加载完整的对象图和派生类型上的导航属性
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-5 加载完整的对象图 问题 你有一个包含许多关联实体的模型,你想在一次查询中, ...
- 《Entity Framework 6 Recipes》中文翻译系列 (26) ------ 第五章 加载实体和导航属性之延缓加载关联实体和在别的LINQ查询操作中使用Include()方法
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-7 在别的LINQ查询操作中使用Include()方法 问题 你有一个LINQ ...
- 《Entity Framework 6 Recipes》中文翻译系列 (27) ------ 第五章 加载实体和导航属性之关联实体过滤、排序、执行聚合操作
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 5-9 关联实体过滤和排序 问题 你有一实体的实例,你想加载应用了过滤和排序的相关 ...
- 《Entity Framework 6 Recipes》翻译系列 (1) -----第一章 开始使用实体框架之历史和框架简述
微软的Entity Framework 受到越来越多人的关注和使用,Entity Framework7.0版本也即将发行.虽然已经开源,可遗憾的是,国内没有关于它的书籍,更不用说好书了,可能是因为EF ...
随机推荐
- 《UML大战需求分析》阅读随笔(六)
在我们做的代码设计中分为系统设计和程序设计.程序设计是系统设计中模拟程序的执行逻辑,定义客户机服务器对象合作的框架的那个部分.程序和事务设计中,作者讲述到程序和事务设计将系统设计制品放在一起,并作为系 ...
- fly bird
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- 【历史】JavaScript和Java没啥关系!————JavaScript简史
文章的开始先上张图: 图片拍摄自北京图书大厦,代表着现在国内应该是绝大部分书店的现状--Javascript书籍放在Java类当中.甚至很多业内人也一直认为Javascript是Java语言在浏览器内 ...
- 弱省互测#2 t2
题意 给两个树,大小分别为n和m,现在两棵树各选一些点(包括1),使得这棵树以1号点为根同构(同构就是每个点的孩子数目相同),求最大的同构树.(n, m<=500) 分析 我们从两棵树中各取出一 ...
- JavaScript笔试必备语句【转】
1. document.write( " "); 输出语句 2.JS中的注释为// 3.传统的HTML文档顺序是:document- >html- >(head,bod ...
- linux下tomcat安装
1.先安装jdk,我们这里用yum进行安装: yum -y install java-1.7.0-openjdk* 确定是否安装成功: java -version 如果显示jdk的版本信息,说明安装成 ...
- 深入浅出HTTP协议(WEB开发和面试必备)
1. 基础概念篇 a.简介 HTTP是Hyper Text Transfer Protocol(超文本传输协议)的缩写.它的发展是万维网协会(World Wide Web Consortium)和 ...
- 谈BFC和haslayout
今天提到BFC和haslayout,就顺带在网上查查资料,总结一下它们. CSS2我们再熟悉不过,当然它里面我们需要掌握的,就是CSS2的选择器和布局,选择器总共31种.避开这个不说,我们说布局. 布 ...
- 封装一些常用的js工具函数-不定时更新(希望大家积极留言,反馈bug^_^)
/*华丽------------------------------------------------------------------------------------------------ ...
- iOS单元格高度计算
// Created by mac on 16/6/29. // Copyright © 2016年 zcc. All rights reserved. // #import "HotCel ...