[翻译 EF Core in Action 1.9] 掀开EF Core的引擎盖看看EF Core内部是如何工作的 Entity Framework Core in Action 掀开EF Core的引擎盖看看EF Core内部是如何工作的 注 书中仅给出了关键代码, 完整示例在 https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

[翻译 EF Core in Action 1.9] 掀开EF Core的引擎盖看看EF Core内部是如何工作的
Entity Framework Core in Action
掀开EF Core的引擎盖看看EF Core内部是如何工作的
注 书中仅给出了关键代码, 完整示例在  https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

Entityframework Core in action是 Jon P smith 所著的关于Entityframework Core 书籍。原版地址. 是除了官方文档外另一个学习EF Core的不错途径, 书中由浅入深的讲解的EF Core的相关知识。因为没有中文版,所以本人对其进行翻译。 预计每两天一篇更新 PS: 翻译难免限于本人水平有不准确的地方,建议英文水平不错的同学直接查看原版,有不足的地方欢迎指正

第一部分目录导航

掀开EF Core的引擎盖看看EF Core内部是如何工作的

创建了MyFirstEfCoreApp应用程序后,你现在可以通过它查看EF Core的工作原理,重点不在于应用程序的代码,而是在读取和写入数据到数据库时EF Core内部会发生什么. 我的目标是让你了解EF Core的工作机制,当你深入研究本书其余部分的命令时,这会很有帮助

注 书中仅给出了关键代码, 完整示例在 https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

数据库建模

在对数据库进行操作之前,EF Core必须进行数据库建模. 数据库建模是EF Core通过实体类和其他EF Core配置来描述数据库的方法. EF Core在所有的数据库访问中使用建立的模型

建模在创建应用程序的DbContext时就开始了,在本例中是AppDbContext(如图1.5所示,在上一篇文章中). 它有属性DbSet

图1.6描述了建模过程的概述,它会帮助你理解EF Core数据库建模的过程. 后续的章节将介绍一系列配置数据库的相关命令,在本文中使用默认配置

[翻译 EF Core in Action 1.9] 掀开EF Core的引擎盖看看EF Core内部是如何工作的
Entity Framework Core in Action
掀开EF Core的引擎盖看看EF Core内部是如何工作的
注 书中仅给出了关键代码, 完整示例在  https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

图1.6展示了EF Core在AppDbContext的建模步骤,下文对此过程进行更详细的说明

  1. EF Core查看DbContext并找到所有公共的DbSet
  2. EF Core查看DbSet
  3. EF Core查找DbSet
  4. 建模过程的最后一个步骤, EF Core运行DbContext的虚方法OnModelCreating, 可以通过重写OnModelCreating方法使用fluent Api进行更多的建模配置,但本例中为了保持示例的简单并没有这样做
  5. EF Core根据收集的信息创建数据库的内部模型,并缓存数据库模式,以便提升访问速度. 在之后的所有的数据库访问中使用此模型

你可能会注意到图1.6并没有展示数据库,因为EF Core构建内部模型时,它不会去查看数据库. 我强调这一点是为了说明构建一个的数据库模型多么重要,如果EF Core认为数据库模型和实际的数据库不匹配,就会出现问题

在你的应用程序中你可以使用EF Core来创建数据库,这会避免出现不匹配的情况. 如果你想要一个良好且高效的数据库,那么在你的代码中编写良好的数据库模型是非常重要的,这样创建的数据库会是高效的. 创建,更新和管理数据库结构是一个很大的主题,将在11章详细介绍

从数据库中读取数据

现在可以访问数据库了. 我们使用List(l)命令,让程序读取数据库并在终端上打印信息. 图1.7显示了输出

[翻译 EF Core in Action 1.9] 掀开EF Core的引擎盖看看EF Core内部是如何工作的
Entity Framework Core in Action
掀开EF Core的引擎盖看看EF Core内部是如何工作的
注 书中仅给出了关键代码, 完整示例在  https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

下面列出代码清单, 用于将所有的图书与作者输出到控制台

[翻译 EF Core in Action 1.9] 掀开EF Core的引擎盖看看EF Core内部是如何工作的
Entity Framework Core in Action
掀开EF Core的引擎盖看看EF Core内部是如何工作的
注 书中仅给出了关键代码, 完整示例在  https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

EF Core使用Linq(语言集成查询)执行它想要执行的命令,使用.net类保存数据

代码清单中粗体显示的两行代码进行了数据库访问. 下面让我们看看EF Core如何使用Linq代码访问数据库并返回数据. 图1.8跟随着这些代码走进EF Core内部,看看不为人知的故事...

[翻译 EF Core in Action 1.9] 掀开EF Core的引擎盖看看EF Core内部是如何工作的
Entity Framework Core in Action
掀开EF Core的引擎盖看看EF Core内部是如何工作的
注 书中仅给出了关键代码, 完整示例在  https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

从数据库中读取数据的过程如下

  1. Linq查询中的db.Books.AsNoTracking().Include(a => a.Author)访问应用程序DbContext的DbSet
  2. 数据库提供程序读取数据后,EF Core通过以下过程放置数据: (a) 创建.NET类的实例 (b) 使用数据库关系链接(外键),通过引用(称为关系修复)将.NET类链接在一起. 结果是一组以正确方式链接的.NET类实例. 在本例中两本书有相同的作者Martin Fowler,因此这两本书的作者属性指向同一个Author类
  3. 由于代码中包含 AsNoTraching, 所以EF Core知道禁止创建跟踪快照. 跟踪快照用于发现数据的变化, 你会在编辑WebUrl的示例中了解这一点. 由于这是一个只读查询,因此禁用跟踪快速会使查询更快

更新数据库

现在使用MyFirstEfCoreApp中的第二个命令update(u)来更新图书Quantum Networking作者的WebUrl列. 如图1.9所示,首先列出所有书籍,会看到最后一本书的作者没有WebUrl. 然后运行命令u,它将要求输入Url. 这时输入 httqs://entangled.moon(这是一个虚构的Url,httpqs-.-),在更新成功后再次列出所有的书籍,这时可以看到Web Url值已经更新

[翻译 EF Core in Action 1.9] 掀开EF Core的引擎盖看看EF Core内部是如何工作的
Entity Framework Core in Action
掀开EF Core的引擎盖看看EF Core内部是如何工作的
注 书中仅给出了关键代码, 完整示例在  https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

代码清单

[翻译 EF Core in Action 1.9] 掀开EF Core的引擎盖看看EF Core内部是如何工作的
Entity Framework Core in Action
掀开EF Core的引擎盖看看EF Core内部是如何工作的
注 书中仅给出了关键代码, 完整示例在  https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

图1.10展示了EF Core内部发生了什么并跟踪其进度,这比上一个read的示例复杂许多, 因此我会给你一些提示

图顶部的读取阶段与上一个读取示例类似,所以应该很熟悉. 在此基础上使用图书的标题做为过滤器载特定的图书. 重要的是第2点: 对数据进行跟踪

在图的下半部分你可以看到EF Core如何将加载的数据与跟踪快照进行比较并找到更改,可以看到只有WebUrl被更新了,它创建了一个SQL命令来只更新该列

[翻译 EF Core in Action 1.9] 掀开EF Core的引擎盖看看EF Core内部是如何工作的
Entity Framework Core in Action
掀开EF Core的引擎盖看看EF Core内部是如何工作的
注 书中仅给出了关键代码, 完整示例在  https://github.com/JonPSmith/EfCoreInAction/tree/Chapter01

图中已经描述了大部分步骤,下面介绍Author的WebUrl列如何更新的详细说明

  1. 应用程序使用LINQ查找包含作者信息的单个图书,EF Core将LINQ查询翻译为SQL命令,读取Title为Quantum Networking的行,返回Book和Author类的实例,因为使用了Single查询,所以还会检查是否只找到一行
  2. LINQ查询中没有AsNoTracking方法,所以该查询是一个具有跟踪的查询,EF Core创建了数据的跟踪快照
  3. 然后代码更改了Book的Author的WebUrl属性. 当调用SaveChanges时, 检测更改阶段会将跟踪的所有类与跟踪快照进行比较. 在这里它会检测到所有已更改的内容. 在本例中主鍵为3的Author实例的WebUrl属性值被更改
  4. 检测到更改后,EF Core将启动事务. 每个数据库更新都以原子单位完成: 更改全部成功或者全部失败. 这非常重要,因为如果仅应用了部分更改,关系数据库可能会发生严重的错误
  5. 更新请求由数据库提供程序转换为SQL命令,如果执行成功则提交事务并返回SaveChanges方法,否则会抛出异常