ASP.NET Core 中的 Razor 页面和 EF Core - 读取相关数据 - 第 6 个教程(共 8 个)Razor Pages with EF Core in ASP.NET Core - Read Related Data - 6 of 8

Contoso University Web 应用演示了如何使用 EF Core 和 Visual Studio 创建 Razor 页面 Web 应用。若要了解系列教程,请参阅。

如果遇到无法解决的问题,请下载已完成的应用,然后对比该代码与按教程所创建的代码。

本教程介绍如何读取和显示相关数据。相关数据为 EF Core 加载到导航属性中的数据。

下图显示了本教程中已完成的页面:

“讲师索引”页

EF Core 可采用多种方式将相关数据加载到实体的导航属性中:

  • 。预先加载是指对查询某类型的实体时一并加载相关实体。读取实体时,会检索其相关数据。此时通常会出现单一联接查询,检索所有必需数据。EF Core 将针对预先加载的某些类型发出多个查询。发布多个查询可能比发布大型的单个查询更为有效。预先加载通过 和 ThenInclude 方法进行指定。

当包含集合导航时,预先加载会发送多个查询:

  • 一个查询用于主查询
  • 一个查询用于加载树中每个集合“边缘”。
    • 使用 Load 的单独查询:可在单独的查询中检索数据,EF Core 会“修复”导航属性。“修复”是指 EF Core 自动填充导航属性。使用 Load 单独查询比预先加载更像是显式加载。

单独查询示例

注意:EF Core 会将导航属性自动“修复”为之前加载到上下文实例中的任何其他实体。即使导航属性的数据非显式包含在内 ,但如果先前加载了部分或所有相关实体,则仍可能填充该属性。

  • 显式加载。首次读取实体时,不检索相关数据。必须编写代码才能在需要时检索相关数据。使用单独查询进行显式加载时,会向数据库发送多个查询。该代码通过显式加载指定要加载的导航属性。使用 Load 方法进行显式加载。例如:

  • 延迟加载已添加到版本 2.1 中的 EF Core。首次读取实体时,不检索相关数据。首次访问导航属性时,会自动检索该导航属性所需的数据。首次访问导航属性时,都会向数据库发送一个查询。

创建“课程”页Create Course pages

Course 实体包括一个带相关 Department 实体的导航属性。

Course.Department

若要显示课程的已分配院系的名称,请执行以下操作:

  • 将相关的 Department 实体加载到 Course.Department 导航属性。
  • 获取 Department 实体的 Name 属性中的名称。

搭建“课程”页的基架Scaffold Course pages

  • 遵循搭建“学生”页的基架中的说明,但以下情况除外:

    • 创建“Pages/Courses”文件夹 。
    • Course 用于模型类。
    • 使用现有的上下文类,而不是新建上下文类。
  • 创建“Pages/Courses”文件夹 。

  • 运行以下命令,搭建“课程”页的基架。

在 Windows 上:

在 Linux 或 macOS 上:

  1. dotnet aspnet-codegenerator razorpage -m Course -dc SchoolContext -udl -outDir Pages/Courses --referenceScriptLibraries
  • 打开 Pages/Courses/Index.cshtml.cs 并检查 OnGetAsync 方法。基架引擎为 Department 导航属性指定了预先加载。Include 方法指定预先加载。

  • 运行应用并选择“课程”链接 。院系列显示 DepartmentID(该项无用)。

显示院系名称Display the department name

使用以下代码更新 Pages/Courses/Index.cshtml.cs:

  1. using ContosoUniversity.Models;
  2. using Microsoft.AspNetCore.Mvc.RazorPages;
  3. using Microsoft.EntityFrameworkCore;
  4. using System.Collections.Generic;
  5. using System.Threading.Tasks;
  6. namespace ContosoUniversity.Pages.Courses
  7. {
  8. public class IndexModel : PageModel
  9. {
  10. private readonly ContosoUniversity.Data.SchoolContext _context;
  11. public IndexModel(ContosoUniversity.Data.SchoolContext context)
  12. {
  13. _context = context;
  14. }
  15. public IList<Course> Courses { get; set; }
  16. public async Task OnGetAsync()
  17. {
  18. Courses = await _context.Courses
  19. .Include(c => c.Department)
  20. .AsNoTracking()
  21. .ToListAsync();
  22. }
  23. }
  24. }

上述代码将 Course 属性更改为 Courses,然后添加 AsNoTracking由于未跟踪返回的实体,因此 AsNoTracking 提升了性能。无需跟踪实体,因为未在当前的上下文中更新这些实体。

使用以下代码更新 Pages/Courses/Index.cshtml 。

  1. @page
  2. @model ContosoUniversity.Pages.Courses.IndexModel
  3. @{
  4. ViewData["Title"] = "Courses";
  5. }
  6. <h1>Courses</h1>
  7. <p>
  8. <a asp-page="Create">Create New</a>
  9. </p>
  10. <table class="table">
  11. <thead>
  12. <tr>
  13. <th>
  14. @Html.DisplayNameFor(model => model.Courses[0].CourseID)
  15. </th>
  16. <th>
  17. @Html.DisplayNameFor(model => model.Courses[0].Title)
  18. </th>
  19. <th>
  20. @Html.DisplayNameFor(model => model.Courses[0].Credits)
  21. </th>
  22. <th>
  23. @Html.DisplayNameFor(model => model.Courses[0].Department)
  24. </th>
  25. <th></th>
  26. </tr>
  27. </thead>
  28. <tbody>
  29. @foreach (var item in Model.Courses)
  30. {
  31. <tr>
  32. <td>
  33. @Html.DisplayFor(modelItem => item.CourseID)
  34. </td>
  35. <td>
  36. @Html.DisplayFor(modelItem => item.Title)
  37. </td>
  38. <td>
  39. @Html.DisplayFor(modelItem => item.Credits)
  40. </td>
  41. <td>
  42. @Html.DisplayFor(modelItem => item.Department.Name)
  43. </td>
  44. <td>
  45. <a asp-page="./Edit" asp-route-id="@item.CourseID">Edit</a> |
  46. <a asp-page="./Details" asp-route-id="@item.CourseID">Details</a> |
  47. <a asp-page="./Delete" asp-route-id="@item.CourseID">Delete</a>
  48. </td>
  49. </tr>
  50. }
  51. </tbody>
  52. </table>

对基架代码进行了以下更改:

  • Course 属性名称更改为了 Courses

  • 添加了显示 CourseID 属性值的“数字”列 。默认情况下,不针对主键进行架构,因为对最终用户而言,它们通常没有意义。但在此情况下主键是有意义的。

  • 更改“院系”列,显示院系名称 。该代码显示已加载到 Department 导航属性中的 Department 实体的 Name 属性:

  1. @Html.DisplayFor(modelItem => item.Department.Name)

运行应用并选择“课程”选项卡,查看包含系名称的列表 。

使用 Select 加载相关数据Loading related data with Select

OnGetAsync 方法使用 Include 方法加载相关数据。Select 方法是只加载所需相关数据的替代方法。对于单个项(如 Department.Name),它使用 SQL INNER JOIN。对于集合,它使用另一个数据库访问,但集合上的 Include 运算符也是如此。

以下代码使用 Select 方法加载相关数据:

  1. public IList<CourseViewModel> CourseVM { get; set; }
  2. public async Task OnGetAsync()
  3. {
  4. CourseVM = await _context.Courses
  5. .Select(p => new CourseViewModel
  6. {
  7. CourseID = p.CourseID,
  8. Title = p.Title,
  9. Credits = p.Credits,
  10. DepartmentName = p.Department.Name
  11. }).ToListAsync();
  12. }

CourseViewModel

  1. public class CourseViewModel
  2. {
  3. public int CourseID { get; set; }
  4. public string Title { get; set; }
  5. public int Credits { get; set; }
  6. public string DepartmentName { get; set; }
  7. }

有关完整示例的信息,请参阅 IndexSelect.cshtml 和 。

创建“讲师”页Create Instructor pages

本节搭建“讲师”页的基架,并向讲师“索引”页添加相关“课程”和“注册”。

“讲师索引”页

该页面通过以下方式读取和显示相关数据:

  • 讲师列表显示 OfficeAssignment 实体(上图中的办公室)的相关数据。InstructorOfficeAssignment 实体之间存在一对零或一的关系。预先加载适用于 OfficeAssignment 实体。需要显示相关数据时,预先加载通常更高效。在此情况下,会显示讲师的办公室分配。
  • 用户选择一名讲师时,显示相关 Course 实体。InstructorCourse 实体之间存在多对多关系。对 Course 实体及其相关的 Department 实体使用预先加载。这种情况下,单独查询可能更有效,因为仅需显示所选讲师的课程。此示例演示如何在位于导航实体内的实体中预先加载这些导航实体。
  • 用户选择一门课程时,会显示 Enrollments 实体的相关数据。上图中显示了学生姓名和成绩。CourseEnrollment 实体之间存在一对多的关系。

“讲师”页显示来自三个不同表格的数据。需要一个视图模型,该模型中包含表示三个表格的三个属性。

使用以下代码创建 SchoolViewModels/InstructorIndexData.cs :

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Threading.Tasks;
  5. namespace ContosoUniversity.Models.SchoolViewModels
  6. {
  7. public class InstructorIndexData
  8. {
  9. public IEnumerable<Instructor> Instructors { get; set; }
  10. public IEnumerable<Course> Courses { get; set; }
  11. public IEnumerable<Enrollment> Enrollments { get; set; }
  12. }
  13. }

搭建“讲师”页的基架Scaffold Instructor pages

  • 遵循中的说明,但以下情况除外:

    • 创建“Pages/Instructors”文件夹 。
    • Instructor 用于模型类。
    • 使用现有的上下文类,而不是新建上下文类。
  • 创建“Pages/Instructors”文件夹 。

  • 运行以下命令,搭建“讲师”页的基架。

在 Windows 上:

  1. dotnet aspnet-codegenerator razorpage -m Instructor -dc SchoolContext -udl -outDir Pages\Instructors --referenceScriptLibraries

在 Linux 或 macOS 上:

  1. dotnet aspnet-codegenerator razorpage -m Instructor -dc SchoolContext -udl -outDir Pages/Instructors --referenceScriptLibraries

若要在更新之前查看已搭建基架的页面的外观,则运行应用并导航到“讲师”页。

使用以下代码更新 Pages/Instructors/Index.cshtml.cs :

  1. using ContosoUniversity.Models;
  2. using ContosoUniversity.Models.SchoolViewModels; // Add VM
  3. using Microsoft.AspNetCore.Mvc.RazorPages;
  4. using Microsoft.EntityFrameworkCore;
  5. using System.Linq;
  6. using System.Threading.Tasks;
  7. namespace ContosoUniversity.Pages.Instructors
  8. {
  9. public class IndexModel : PageModel
  10. {
  11. private readonly ContosoUniversity.Data.SchoolContext _context;
  12. public IndexModel(ContosoUniversity.Data.SchoolContext context)
  13. {
  14. _context = context;
  15. }
  16. public InstructorIndexData InstructorData { get; set; }
  17. public int InstructorID { get; set; }
  18. public int CourseID { get; set; }
  19. public async Task OnGetAsync(int? id, int? courseID)
  20. InstructorData = new InstructorIndexData();
  21. InstructorData.Instructors = await _context.Instructors
  22. .Include(i => i.OfficeAssignment)
  23. .Include(i => i.CourseAssignments)
  24. .ThenInclude(i => i.Course)
  25. .ThenInclude(i => i.Department)
  26. .Include(i => i.CourseAssignments)
  27. .ThenInclude(i => i.Course)
  28. .ThenInclude(i => i.Enrollments)
  29. .ThenInclude(i => i.Student)
  30. .AsNoTracking()
  31. .OrderBy(i => i.LastName)
  32. .ToListAsync();
  33. if (id != null)
  34. {
  35. InstructorID = id.Value;
  36. Instructor instructor = InstructorData.Instructors
  37. .Where(i => i.ID == id.Value).Single();
  38. InstructorData.Courses = instructor.CourseAssignments.Select(s => s.Course);
  39. }
  40. if (courseID != null)
  41. {
  42. CourseID = courseID.Value;
  43. var selectedCourse = InstructorData.Courses
  44. .Where(x => x.CourseID == courseID).Single();
  45. InstructorData.Enrollments = selectedCourse.Enrollments;
  46. }
  47. }
  48. }
  49. }

OnGetAsync 方法接受所选讲师 ID 的可选路由数据。

检查 Pages/Instructors/Index.cshtml.cs 文件中的查询 :

  1. InstructorData.Instructors = await _context.Instructors
  2. .Include(i => i.OfficeAssignment)
  3. .Include(i => i.CourseAssignments)
  4. .ThenInclude(i => i.Course)
  5. .ThenInclude(i => i.Department)
  6. .Include(i => i.CourseAssignments)
  7. .ThenInclude(i => i.Course)
  8. .ThenInclude(i => i.Enrollments)
  9. .ThenInclude(i => i.Student)
  10. .AsNoTracking()
  11. .OrderBy(i => i.LastName)
  12. .ToListAsync();

代码指定以下导航属性的预先加载:

  • Instructor.OfficeAssignment
  • Instructor.CourseAssignments
    • CourseAssignments.Course
      • Course.Department
      • Course.Enrollments
        • Enrollment.Student

注意 CourseAssignments 和 对 IncludeThenInclude 方法的重复使用。若要指定 Course 实体的两个导航属性的预先加载,则这种重复使用是必要的。

选择讲师时 (id != null),将执行以下代码。

  1. if (id != null)
  2. {
  3. InstructorID = id.Value;
  4. Instructor instructor = InstructorData.Instructors
  5. .Where(i => i.ID == id.Value).Single();
  6. InstructorData.Courses = instructor.CourseAssignments.Select(s => s.Course);
  7. }

从视图模型中的讲师列表检索所选讲师。向视图模型的 Courses 属性加载来自讲师 CourseAssignments 导航属性的 Course 实体。

Where 方法返回一个集合。但在本例中,筛选器将选择单个实体。因此,调用 Single 方法将集合转换为单个 Instructor 实体。Instructor 实体提供对 CourseAssignments 属性的访问。CourseAssignments 提供对相关 Course 实体的访问。

当集合仅包含一个项时,集合使用 Single 方法。如果集合为空或包含多个项,Single 方法会引发异常。还可使用 SingleOrDefault,该方式在集合为空时返回默认值(本例中为 null)。

选中课程时,视图模型的 Enrollments 属性将填充以下代码:

  1. if (courseID != null)
  2. {
  3. CourseID = courseID.Value;
  4. var selectedCourse = InstructorData.Courses
  5. .Where(x => x.CourseID == courseID).Single();
  6. InstructorData.Enrollments = selectedCourse.Enrollments;
  7. }

更新“讲师索引”页Update the instructors Index page

使用以下代码更新 Pages/Instructors/Index.cshtml 。

  1. @page "{id:int?}"
  2. @model ContosoUniversity.Pages.Instructors.IndexModel
  3. @{
  4. ViewData["Title"] = "Instructors";
  5. }
  6. <h2>Instructors</h2>
  7. <p>
  8. <a asp-page="Create">Create New</a>
  9. </p>
  10. <table class="table">
  11. <thead>
  12. <tr>
  13. <th>Last Name</th>
  14. <th>First Name</th>
  15. <th>Hire Date</th>
  16. <th>Office</th>
  17. <th>Courses</th>
  18. <th></th>
  19. </tr>
  20. </thead>
  21. <tbody>
  22. @foreach (var item in Model.InstructorData.Instructors)
  23. {
  24. string selectedRow = "";
  25. if (item.ID == Model.InstructorID)
  26. {
  27. selectedRow = "table-success";
  28. }
  29. <tr class="@selectedRow">
  30. <td>
  31. @Html.DisplayFor(modelItem => item.LastName)
  32. </td>
  33. <td>
  34. @Html.DisplayFor(modelItem => item.FirstMidName)
  35. </td>
  36. <td>
  37. @Html.DisplayFor(modelItem => item.HireDate)
  38. </td>
  39. <td>
  40. @if (item.OfficeAssignment != null)
  41. {
  42. @item.OfficeAssignment.Location
  43. }
  44. </td>
  45. <td>
  46. @{
  47. foreach (var course in item.CourseAssignments)
  48. {
  49. @course.Course.CourseID @: @course.Course.Title <br />
  50. }
  51. }
  52. </td>
  53. <td>
  54. <a asp-page="./Index" asp-route-id="@item.ID">Select</a> |
  55. <a asp-page="./Edit" asp-route-id="@item.ID">Edit</a> |
  56. <a asp-page="./Details" asp-route-id="@item.ID">Details</a> |
  57. <a asp-page="./Delete" asp-route-id="@item.ID">Delete</a>
  58. </td>
  59. </tr>
  60. }
  61. </tbody>
  62. </table>
  63. @if (Model.InstructorData.Courses != null)
  64. {
  65. <h3>Courses Taught by Selected Instructor</h3>
  66. <table class="table">
  67. <tr>
  68. <th></th>
  69. <th>Number</th>
  70. <th>Title</th>
  71. <th>Department</th>
  72. </tr>
  73. @foreach (var item in Model.InstructorData.Courses)
  74. {
  75. string selectedRow = "";
  76. if (item.CourseID == Model.CourseID)
  77. {
  78. selectedRow = "table-success";
  79. }
  80. <tr class="@selectedRow">
  81. <td>
  82. <a asp-page="./Index" asp-route-courseID="@item.CourseID">Select</a>
  83. </td>
  84. <td>
  85. @item.CourseID
  86. </td>
  87. <td>
  88. @item.Title
  89. </td>
  90. <td>
  91. @item.Department.Name
  92. </td>
  93. </tr>
  94. }
  95. </table>
  96. }
  97. @if (Model.InstructorData.Enrollments != null)
  98. {
  99. <h3>
  100. Students Enrolled in Selected Course
  101. </h3>
  102. <table class="table">
  103. <tr>
  104. <th>Name</th>
  105. <th>Grade</th>
  106. </tr>
  107. @foreach (var item in Model.InstructorData.Enrollments)
  108. {
  109. <tr>
  110. <td>
  111. @item.Student.FullName
  112. </td>
  113. <td>
  114. @Html.DisplayFor(modelItem => item.Grade)
  115. </td>
  116. </tr>
  117. }
  118. </table>
  119. }

上面的代码执行以下更改:

  • page 指令从 @page 更新为 @page "{id:int?}""{id:int?}" 是一个路由模板。路由模板将 URL 中的整数查询字符串更改为路由数据。例如,单击仅具有 @page 指令的讲师的“选择”链接将生成如下 URL :

如果页面指令为 @page "{id:int?}" 时,则 URL 为:

https://localhost:5001/Instructors/2

  • 添加仅在 item.OfficeAssignment 不为 null 时才显示 item.OfficeAssignment.Location 的“办公室”列 。由于这是一对零或一的关系,因此可能没有相关的 OfficeAssignment 实体。
  • 添加显示每位讲师所授课程的“课程”列 。有关此 razor 语法的详细信息,请参阅。

  • 添加向所选讲师和课程的 tr 元素中动态添加 class="success" 的代码。此时会使用 Bootstrap 类为所选行设置背景色。

  1. string selectedRow = "";
  2. if (item.CourseID == Model.CourseID)
  3. {
  4. selectedRow = "success";
  5. }
  6. <tr class="@selectedRow">
  • 添加标记为“选择”的新的超链接 。该链接将所选讲师的 ID 发送给 Index 方法并设置背景色。
  1. <a asp-action="Index" asp-route-id="@item.ID">Select</a> |
  • 添加所选课程的学生注册表。

运行应用并选择“讲师”选项卡 。该页显示来自相关 OfficeAssignment 实体的 Location(办公室)。如果 OfficeAssignment 为 NULL,则显示空白表格单元格。

单击“选择”链接,选择讲师 。显示行样式更改和分配给该讲师的课程。

选择一门课程,查看已注册的学生及其成绩列表。

使用 Single 方法Using Single

Single 方法可在 Where 条件中进行传递,无需分别调用 Where 方法:

  1. public async Task OnGetAsync(int? id, int? courseID)
  2. {
  3. InstructorData = new InstructorIndexData();
  4. InstructorData.Instructors = await _context.Instructors
  5. .Include(i => i.OfficeAssignment)
  6. .Include(i => i.CourseAssignments)
  7. .ThenInclude(i => i.Course)
  8. .ThenInclude(i => i.Department)
  9. .Include(i => i.CourseAssignments)
  10. .ThenInclude(i => i.Course)
  11. .ThenInclude(i => i.Enrollments)
  12. .ThenInclude(i => i.Student)
  13. .AsNoTracking()
  14. .OrderBy(i => i.LastName)
  15. .ToListAsync();
  16. if (id != null)
  17. {
  18. InstructorID = id.Value;
  19. Instructor instructor = InstructorData.Instructors.Single(
  20. i => i.ID == id.Value);
  21. InstructorData.Courses = instructor.CourseAssignments.Select(
  22. s => s.Course);
  23. }
  24. if (courseID != null)
  25. {
  26. CourseID = courseID.Value;
  27. InstructorData.Enrollments = InstructorData.Courses.Single(
  28. x => x.CourseID == courseID).Enrollments;
  29. }
  30. }

Single 与 Where 条件的配合使用与个人偏好相关。相较于使用 Where 方法,它没有提供任何优势。

当前代码为 EnrollmentsStudents 指定预先加载:

  1. InstructorData.Instructors = await _context.Instructors
  2. .Include(i => i.OfficeAssignment)
  3. .Include(i => i.CourseAssignments)
  4. .ThenInclude(i => i.Course)
  5. .ThenInclude(i => i.Department)
  6. .Include(i => i.CourseAssignments)
  7. .ThenInclude(i => i.Course)
  8. .ThenInclude(i => i.Enrollments)
  9. .ThenInclude(i => i.Student)
  10. .AsNoTracking()
  11. .OrderBy(i => i.LastName)
  12. .ToListAsync();

假设用户几乎不希望课程中显示注册情况。在此情况下,可仅在请求时加载注册数据进行优化。在本部分中,会更新 OnGetAsync 以使用 EnrollmentsStudents 的显式加载。

使用以下代码更新 Pages/Instructors/Index.cshtml.cs 。

  1. using ContosoUniversity.Models;
  2. using ContosoUniversity.Models.SchoolViewModels; // Add VM
  3. using Microsoft.AspNetCore.Mvc.RazorPages;
  4. using Microsoft.EntityFrameworkCore;
  5. using System.Linq;
  6. using System.Threading.Tasks;
  7. namespace ContosoUniversity.Pages.Instructors
  8. {
  9. public class IndexModel : PageModel
  10. {
  11. private readonly ContosoUniversity.Data.SchoolContext _context;
  12. public IndexModel(ContosoUniversity.Data.SchoolContext context)
  13. {
  14. _context = context;
  15. }
  16. public InstructorIndexData InstructorData { get; set; }
  17. public int InstructorID { get; set; }
  18. public int CourseID { get; set; }
  19. public async Task OnGetAsync(int? id, int? courseID)
  20. {
  21. InstructorData = new InstructorIndexData();
  22. InstructorData.Instructors = await _context.Instructors
  23. .Include(i => i.OfficeAssignment)
  24. .Include(i => i.CourseAssignments)
  25. .ThenInclude(i => i.Course)
  26. .ThenInclude(i => i.Department)
  27. //.Include(i => i.CourseAssignments)
  28. // .ThenInclude(i => i.Course)
  29. // .ThenInclude(i => i.Enrollments)
  30. // .ThenInclude(i => i.Student)
  31. //.AsNoTracking()
  32. .OrderBy(i => i.LastName)
  33. .ToListAsync();
  34. if (id != null)
  35. {
  36. InstructorID = id.Value;
  37. Instructor instructor = InstructorData.Instructors
  38. .Where(i => i.ID == id.Value).Single();
  39. InstructorData.Courses = instructor.CourseAssignments.Select(s => s.Course);
  40. }
  41. if (courseID != null)
  42. {
  43. CourseID = courseID.Value;
  44. var selectedCourse = InstructorData.Courses
  45. .Where(x => x.CourseID == courseID).Single();
  46. await _context.Entry(selectedCourse).Collection(x => x.Enrollments).LoadAsync();
  47. foreach (Enrollment enrollment in selectedCourse.Enrollments)
  48. {
  49. await _context.Entry(enrollment).Reference(x => x.Student).LoadAsync();
  50. }
  51. InstructorData.Enrollments = selectedCourse.Enrollments;
  52. }
  53. }
  54. }
  55. }

上述代码取消针对注册和学生数据的 ThenInclude 方法调用 。如果已选中课程,则显式加载的代码会检索:

  • 所选课程的 Enrollment 实体。
  • 每个 EnrollmentStudent 实体。

注意,上述代码注释掉了 .AsNoTracking()对于跟踪的实体,仅可显式加载导航属性。

测试应用。对用户而言,该应用的行为与上一版本相同。

后续步骤Next steps

下一个教程将介绍如何更新相关数据。

上一个教程

在本教程中,将读取和显示相关数据。相关数据为 EF Core 加载到导航属性中的数据。

如果遇到无法解决的问题,请下载或查看已完成的应用

下图显示了本教程中已完成的页面:

“课程索引”页

相关数据的预先加载、显式加载和延迟加载Eager, explicit, and lazy Loading of related data

EF Core 可采用多种方式将相关数据加载到实体的导航属性中:

  • 。预先加载是指对查询某类型的实体时一并加载相关实体。读取实体时,会检索其相关数据。此时通常会出现单一联接查询,检索所有必需数据。EF Core 将针对预先加载的某些类型发出多个查询。与存在单一查询的 EF6 中的某些查询相比,发出多个查询可能更有效。预先加载通过 IncludeThenInclude 方法进行指定。

预先加载示例

当包含集合导航时,预先加载会发送多个查询:

  • 一个查询用于主查询
  • 一个查询用于加载树中每个集合“边缘”。
    • 使用 Load 的单独查询:可在单独的查询中检索数据,EF Core 会“修复”导航属性。“修复”是指 EF Core 自动填充导航属性。使用 Load 单独查询比预先加载更像是显式加载。

注意:EF Core 会将导航属性自动“修复”为之前加载到上下文实例中的任何其他实体。即使导航属性的数据非显式包含在内 ,但如果先前加载了部分或所有相关实体,则仍可能填充该属性。

  • 显式加载。首次读取实体时,不检索相关数据。必须编写代码才能在需要时检索相关数据。使用单独查询进行显式加载时,会向数据库发送多个查询。该代码通过显式加载指定要加载的导航属性。使用 Load 方法进行显式加载。例如:

显式加载示例

  • 延迟加载已添加到版本 2.1 中的 EF Core。首次读取实体时,不检索相关数据。首次访问导航属性时,会自动检索该导航属性所需的数据。首次访问导航属性时,都会向数据库发送一个查询。

  • Select 运算符仅加载所需的相关数据。

创建显示院系名称的“课程”页Create a Course page that displays department name

课程实体包括一个带 Department 实体的导航属性。Department 实体包含要分配课程的院系。

要在课程列表中显示已分配院系的名称:

  • Department 实体中获取 Name 属性。
  • Department 实体来自于 Course.Department 导航属性。

为课程模型创建基架Scaffold the Course model

按照为“学生”模型搭建基架中的说明操作,并对模型类使用 Course

运行下面的命令:

  1. dotnet aspnet-codegenerator razorpage -m Course -dc SchoolContext -udl -outDir Pages\Courses --referenceScriptLibraries

上述命令为 Course 模型创建基架。在 Visual Studio 中打开项目。

打开 Pages/Courses/Index.cshtml.cs 并检查 OnGetAsync 方法。基架引擎为 Department 导航属性指定了预先加载。Include 方法指定预先加载。

运行应用并选择“课程”链接 。院系列显示 DepartmentID(该项无用)。

使用以下代码更新 OnGetAsync 方法:

  1. public async Task OnGetAsync()
  2. {
  3. Course = await _context.Courses
  4. .Include(c => c.Department)
  5. .AsNoTracking()
  6. .ToListAsync();
  7. }

上述代码添加了 AsNoTracking由于未跟踪返回的实体,因此 AsNoTracking 提升了性能。未跟踪实体,因为未在当前上下文中更新这些实体。

使用以下突出显示的标记更新 Pages/Courses/Index.cshtml :

  1. @page
  2. @model ContosoUniversity.Pages.Courses.IndexModel
  3. @{
  4. ViewData["Title"] = "Courses";
  5. }
  6. <h2>Courses</h2>
  7. <p>
  8. <a asp-page="Create">Create New</a>
  9. </p>
  10. <table class="table">
  11. <thead>
  12. <tr>
  13. <th>
  14. @Html.DisplayNameFor(model => model.Course[0].CourseID)
  15. </th>
  16. <th>
  17. @Html.DisplayNameFor(model => model.Course[0].Title)
  18. </th>
  19. <th>
  20. @Html.DisplayNameFor(model => model.Course[0].Credits)
  21. </th>
  22. <th>
  23. @Html.DisplayNameFor(model => model.Course[0].Department)
  24. </th>
  25. <th></th>
  26. </tr>
  27. </thead>
  28. <tbody>
  29. @foreach (var item in Model.Course)
  30. {
  31. <tr>
  32. <td>
  33. @Html.DisplayFor(modelItem => item.CourseID)
  34. </td>
  35. <td>
  36. @Html.DisplayFor(modelItem => item.Title)
  37. </td>
  38. <td>
  39. @Html.DisplayFor(modelItem => item.Credits)
  40. </td>
  41. <td>
  42. @Html.DisplayFor(modelItem => item.Department.Name)
  43. </td>
  44. <td>
  45. <a asp-page="./Edit" asp-route-id="@item.CourseID">Edit</a> |
  46. <a asp-page="./Details" asp-route-id="@item.CourseID">Details</a> |
  47. <a asp-page="./Delete" asp-route-id="@item.CourseID">Delete</a>
  48. </td>
  49. </tr>
  50. }
  51. </tbody>
  52. </table>

对基架代码进行了以下更改:

  • 将标题从“索引”更改为“课程”。

  • 添加了显示 CourseID 属性值的“数字”列 。默认情况下,不针对主键进行架构,因为对最终用户而言,它们通常没有意义。但在此情况下主键是有意义的。

  • 更改“院系”列,显示院系名称 。该代码显示已加载到 Department 导航属性中的 Department 实体的 Name 属性:

@Html.DisplayFor(modelItem => item.Department.Name)

运行应用并选择“课程”选项卡,查看包含系名称的列表 。

“课程索引”页

OnGetAsync 方法使用 Include 方法加载相关数据:

public async Task OnGetAsync()
{
    Course = await _context.Courses
        .Include(c => c.Department)
        .AsNoTracking()
        .ToListAsync();
}

Select 运算符仅加载所需的相关数据。对于单个项(如 Department.Name),它使用 SQL INNER JOIN。对于集合,它使用另一个数据库访问,但集合上的 Include 运算符也是如此。

以下代码使用 Select 方法加载相关数据:

public IList<CourseViewModel> CourseVM { get; set; }

public async Task OnGetAsync()
{
    CourseVM = await _context.Courses
            .Select(p => new CourseViewModel
            {
                CourseID = p.CourseID,
                Title = p.Title,
                Credits = p.Credits,
                DepartmentName = p.Department.Name
            }).ToListAsync();
}

CourseViewModel

public class CourseViewModel
{
    public int CourseID { get; set; }
    public string Title { get; set; }
    public int Credits { get; set; }
    public string DepartmentName { get; set; }
}

有关完整示例的信息,请参阅 和 IndexSelect.cshtml.cs

在本部分中,将创建“讲师”页。

该页面通过以下方式读取和显示相关数据:

  • 讲师列表显示 OfficeAssignment 实体(上图中的办公室)的相关数据。InstructorOfficeAssignment 实体之间存在一对零或一的关系。预先加载适用于 OfficeAssignment 实体。需要显示相关数据时,预先加载通常更高效。在此情况下,会显示讲师的办公室分配。
  • 当用户选择一名讲师(上图中的 Harui)时,显示相关的 Course 实体。InstructorCourse 实体之间存在多对多关系。对 Course 实体及其相关的 Department 实体使用预先加载。这种情况下,单独查询可能更有效,因为仅需显示所选讲师的课程。此示例演示如何在位于导航实体内的实体中预先加载这些导航实体。
  • 当用户选择一门课程(上图中的化学)时,显示 Enrollments 实体的相关数据。上图中显示了学生姓名和成绩。CourseEnrollment 实体之间存在一对多的关系。

创建“讲师索引”视图的视图模型Create a view model for the Instructor Index view

“讲师”页显示来自三个不同表格的数据。创建一个视图模型,该模型中包含表示三个表格的三个实体。

在 SchoolViewModels 文件夹中,使用以下代码创建 InstructorIndexData.cs :

using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;

namespace ContosoUniversity.Models.SchoolViewModels
{
    public class InstructorIndexData
    {
        public IEnumerable<Instructor> Instructors { get; set; }
        public IEnumerable<Course> Courses { get; set; }
        public IEnumerable<Enrollment> Enrollments { get; set; }
    }
}

为讲师模型创建基架Scaffold the Instructor model

按照为“学生”模型搭建基架中的说明操作,并对模型类使用 Instructor

运行下面的命令:

dotnet aspnet-codegenerator razorpage -m Instructor -dc SchoolContext -udl -outDir Pages\Instructors --referenceScriptLibraries

上述命令为 Instructor 模型创建基架。运行应用并导航到“讲师”页。

将 Pages/Instructors/Index.cshtml.cs 替换为以下代码:

OnGetAsync 方法接受所选讲师 ID 的可选路由数据。

检查 Pages/Instructors/Index.cshtml.cs 文件中的查询 :

Instructor.Instructors = await _context.Instructors
      .Include(i => i.OfficeAssignment)
      .Include(i => i.CourseAssignments)
        .ThenInclude(i => i.Course)
      .AsNoTracking()
      .OrderBy(i => i.LastName)
      .ToListAsync();

查询包括两项内容:

  • OfficeAssignment:在中显示。
  • CourseAssignments:课程的教学内容。

更新“讲师索引”页Update the instructors Index page

使用以下标记更新 Pages/Instructors/Index.cshtml :

@page "{id:int?}"
@model ContosoUniversity.Pages.Instructors.IndexModel

@{
    ViewData["Title"] = "Instructors";
}

<h2>Instructors</h2>

<p>
    <a asp-page="Create">Create New</a>
</p>
<table class="table">
    <thead>
        <tr>
            <th>Last Name</th>
            <th>First Name</th>
            <th>Hire Date</th>
            <th>Office</th>
            <th>Courses</th>
            <th></th>
        </tr>
    </thead>
    <tbody>
        @foreach (var item in Model.Instructor.Instructors)
        {
            string selectedRow = "";
            if (item.ID == Model.InstructorID)
            {
                selectedRow = "success";
            }
            <tr class="@selectedRow">
                <td>
                    @Html.DisplayFor(modelItem => item.LastName)
                </td>
                <td>
                    @Html.DisplayFor(modelItem => item.FirstMidName)
                </td>
                <td>
                    @Html.DisplayFor(modelItem => item.HireDate)
                </td>
                <td>
                    @if (item.OfficeAssignment != null)
                    {
                        @item.OfficeAssignment.Location
                    }
                </td>
                <td>
                    @{
                        foreach (var course in item.CourseAssignments)
                        {
                            @course.Course.CourseID @:  @course.Course.Title <br />
                        }
                    }
                </td>
                <td>
                    <a asp-page="./Index" asp-route-id="@item.ID">Select</a> |
                    <a asp-page="./Edit" asp-route-id="@item.ID">Edit</a> |
                    <a asp-page="./Details" asp-route-id="@item.ID">Details</a> |
                    <a asp-page="./Delete" asp-route-id="@item.ID">Delete</a>
                </td>
            </tr>
        }
    </tbody>
</table>

上述标记进行以下更改:

  • page 指令从 @page 更新为 @page "{id:int?}""{id:int?}" 是一个路由模板。路由模板将 URL 中的整数查询字符串更改为路由数据。例如,单击仅具有 @page 指令的讲师的“选择”链接将生成如下 URL :

当页面指令是 @page "{id:int?}" 时,之前的 URL 为:

http://localhost:1234/Instructors/2

  • 页标题为“讲师” 。

  • 添加了仅在 item.OfficeAssignment 不为 null 时才显示 item.OfficeAssignment.Location 的“办公室”列 。由于这是一对零或一的关系,因此可能没有相关的 OfficeAssignment 实体。

@if (item.OfficeAssignment != null)
{
    @item.OfficeAssignment.Location
}
  • 添加了显示每位讲师所授课程的“课程”列 。有关此 razor 语法的详细信息,请参阅。

  • 添加了向所选讲师的 tr 元素中动态添加 class="success" 的代码。此时会使用 Bootstrap 类为所选行设置背景色。

string selectedRow = "";
if (item.CourseID == Model.CourseID)
{
    selectedRow = "success";
}
<tr class="@selectedRow">
  • 添加了标记为“选择”的新的超链接 。该链接将所选讲师的 ID 发送给 Index 方法并设置背景色。
<a asp-action="Index" asp-route-id="@item.ID">Select</a> |

运行应用并选择“讲师”选项卡 。该页显示来自相关 OfficeAssignment 实体的 Location(办公室)。如果 OfficeAssignment` 为 NULL,则显示空白表格单元格。

单击“选择” 链接。随即更改行样式。

将 Pages/Instructors/Index.cshtml.cs 中的 OnGetAsync 方法替换为以下代码:

public async Task OnGetAsync(int? id, int? courseID)
{
    Instructor = new InstructorIndexData();
    Instructor.Instructors = await _context.Instructors
          .Include(i => i.OfficeAssignment)
          .Include(i => i.CourseAssignments)
            .ThenInclude(i => i.Course)
                .ThenInclude(i => i.Department)
          .AsNoTracking()
          .OrderBy(i => i.LastName)
          .ToListAsync();

    if (id != null)
    {
        InstructorID = id.Value;
        Instructor instructor = Instructor.Instructors.Where(
            i => i.ID == id.Value).Single();
        Instructor.Courses = instructor.CourseAssignments.Select(s => s.Course);
    }

    if (courseID != null)
    {
        CourseID = courseID.Value;
        Instructor.Enrollments = Instructor.Courses.Where(
            x => x.CourseID == courseID).Single().Enrollments;
    }
}

添加 public int CourseID { get; set; }

public class IndexModel : PageModel
{
    private readonly ContosoUniversity.Data.SchoolContext _context;

    public IndexModel(ContosoUniversity.Data.SchoolContext context)
    {
        _context = context;
    }

    public InstructorIndexData Instructor { get; set; }
    public int InstructorID { get; set; }
    public int CourseID { get; set; }

    public async Task OnGetAsync(int? id, int? courseID)
    {
        Instructor = new InstructorIndexData();
        Instructor.Instructors = await _context.Instructors
              .Include(i => i.OfficeAssignment)
              .Include(i => i.CourseAssignments)
                .ThenInclude(i => i.Course)
                    .ThenInclude(i => i.Department)
              .AsNoTracking()
              .OrderBy(i => i.LastName)
              .ToListAsync();

        if (id != null)
        {
            InstructorID = id.Value;
            Instructor instructor = Instructor.Instructors.Where(
                i => i.ID == id.Value).Single();
            Instructor.Courses = instructor.CourseAssignments.Select(s => s.Course);
        }

        if (courseID != null)
        {
            CourseID = courseID.Value;
            Instructor.Enrollments = Instructor.Courses.Where(
                x => x.CourseID == courseID).Single().Enrollments;
        }
    }

检查更新后的查询:

Instructor.Instructors = await _context.Instructors
      .Include(i => i.OfficeAssignment)
      .Include(i => i.CourseAssignments)
        .ThenInclude(i => i.Course)
            .ThenInclude(i => i.Department)
      .AsNoTracking()
      .OrderBy(i => i.LastName)
      .ToListAsync();

先前查询添加了 Department 实体。

选择讲师时 (id != null),将执行以下代码。从视图模型中的讲师列表检索所选讲师。向视图模型的 Courses 属性加载来自讲师 CourseAssignments 导航属性的 Course 实体。

if (id != null)
{
    InstructorID = id.Value;
    Instructor instructor = Instructor.Instructors.Where(
        i => i.ID == id.Value).Single();
    Instructor.Courses = instructor.CourseAssignments.Select(s => s.Course);
}

Where 方法返回一个集合。在前面的 Where 方法中,仅返回单个 Instructor 实体。Single 方法将集合转换为单个 Instructor 实体。Instructor 实体提供对 CourseAssignments 属性的访问。CourseAssignments 提供对相关 Course 实体的访问。

讲师-课程 m:M

当集合仅包含一个项时,集合使用 Single 方法。如果集合为空或包含多个项,Single 方法会引发异常。还可使用 SingleOrDefault,该方式在集合为空时返回默认值(本例中为 null)。在空集合上使用 SingleOrDefault

  • 引发异常(因为尝试在空引用上找到 Courses 属性)。
  • 异常信息不太能清楚指出问题原因。

选中课程时,视图模型的 Enrollments 属性将填充以下代码:

if (courseID != null)
{
    CourseID = courseID.Value;
    Instructor.Enrollments = Instructor.Courses.Where(
        x => x.CourseID == courseID).Single().Enrollments;
}

在 Pages/Instructors/Index.cshtml Razor 页面末尾添加以下标记 :

                    <a asp-page="./Delete" asp-route-id="@item.ID">Delete</a>
                </td>
            </tr>
        }
    </tbody>
</table>

@if (Model.Instructor.Courses != null)
{
    <h3>Courses Taught by Selected Instructor</h3>
    <table class="table">
        <tr>
            <th></th>
            <th>Number</th>
            <th>Title</th>
            <th>Department</th>
        </tr>

        @foreach (var item in Model.Instructor.Courses)
        {
            string selectedRow = "";
            if (item.CourseID == Model.CourseID)
            {
                selectedRow = "success";
            }
            <tr class="@selectedRow">
                <td>
                    <a asp-page="./Index" asp-route-courseID="@item.CourseID">Select</a>
                </td>
                <td>
                    @item.CourseID
                </td>
                <td>
                    @item.Title
                </td>
                <td>
                    @item.Department.Name
                </td>
            </tr>
        }

    </table>
}

上述标记显示选中某讲师时与该讲师相关的课程列表。

测试应用。单击讲师页面上的“选择” 链接。

显示学生数据Show student data

在本部分中,更新应用以显示所选课程的学生数据。

使用以下代码在 Pages/Instructors/Index.cshtml.cs 中更新 OnGetAsync 方法中的查询:

Instructor.Instructors = await _context.Instructors
      .Include(i => i.OfficeAssignment)                 
      .Include(i => i.CourseAssignments)
        .ThenInclude(i => i.Course)
            .ThenInclude(i => i.Department)
        .Include(i => i.CourseAssignments)
            .ThenInclude(i => i.Course)
                .ThenInclude(i => i.Enrollments)
                    .ThenInclude(i => i.Student)
      .AsNoTracking()
      .OrderBy(i => i.LastName)
      .ToListAsync();

更新 Pages/Instructors/Index.cshtml 。在文件末尾添加以下标记:


@if (Model.Instructor.Enrollments != null)
{
    <h3>
        Students Enrolled in Selected Course
    </h3>
    <table class="table">
        <tr>
            <th>Name</th>
            <th>Grade</th>
        </tr>
        @foreach (var item in Model.Instructor.Enrollments)
        {
            <tr>
                <td>
                    @item.Student.FullName
                </td>
                <td>
                    @Html.DisplayFor(modelItem => item.Grade)
                </td>
            </tr>
        }
    </table>
}

上述标记显示已注册所选课程的学生列表。

刷新页面并选择讲师。选择一门课程,查看已注册的学生及其成绩列表。

使用 Single 方法Using Single

Single 方法可在 Where 条件中进行传递,无需分别调用 Where 方法:

public async Task OnGetAsync(int? id, int? courseID)
{
    Instructor = new InstructorIndexData();

    Instructor.Instructors = await _context.Instructors
          .Include(i => i.OfficeAssignment)
          .Include(i => i.CourseAssignments)
            .ThenInclude(i => i.Course)
                .ThenInclude(i => i.Department)
            .Include(i => i.CourseAssignments)
                .ThenInclude(i => i.Course)
                    .ThenInclude(i => i.Enrollments)
                        .ThenInclude(i => i.Student)
          .AsNoTracking()
          .OrderBy(i => i.LastName)
          .ToListAsync();

    if (id != null)
    {
        InstructorID = id.Value;
        Instructor instructor = Instructor.Instructors.Single(
            i => i.ID == id.Value);
        Instructor.Courses = instructor.CourseAssignments.Select(
            s => s.Course);
    }

    if (courseID != null)
    {
        CourseID = courseID.Value;
        Instructor.Enrollments = Instructor.Courses.Single(
            x => x.CourseID == courseID).Enrollments;
    }
}

使用 Where 时,前面的 Single 方法不适用。一些开发人员更喜欢 Single 方法样式。

显式加载Explicit loading

当前代码为 EnrollmentsStudents 指定预先加载:

假设用户几乎不希望课程中显示注册情况。在此情况下,可仅在请求时加载注册数据进行优化。在本部分中,会更新 OnGetAsync 以使用 EnrollmentsStudents 的显式加载。

使用以下代码更新 OnGetAsync

public async Task OnGetAsync(int? id, int? courseID)
{
    Instructor = new InstructorIndexData();
    Instructor.Instructors = await _context.Instructors
          .Include(i => i.OfficeAssignment)                 
          .Include(i => i.CourseAssignments)
            .ThenInclude(i => i.Course)
                .ThenInclude(i => i.Department)
            //.Include(i => i.CourseAssignments)
            //    .ThenInclude(i => i.Course)
            //        .ThenInclude(i => i.Enrollments)
            //            .ThenInclude(i => i.Student)
         // .AsNoTracking()
          .OrderBy(i => i.LastName)
          .ToListAsync();


    if (id != null)
    {
        InstructorID = id.Value;
        Instructor instructor = Instructor.Instructors.Where(
            i => i.ID == id.Value).Single();
        Instructor.Courses = instructor.CourseAssignments.Select(s => s.Course);
    }

    if (courseID != null)
    {
        CourseID = courseID.Value;
        var selectedCourse = Instructor.Courses.Where(x => x.CourseID == courseID).Single();
        await _context.Entry(selectedCourse).Collection(x => x.Enrollments).LoadAsync();
        foreach (Enrollment enrollment in selectedCourse.Enrollments)
        {
            await _context.Entry(enrollment).Reference(x => x.Student).LoadAsync();
        }
        Instructor.Enrollments = selectedCourse.Enrollments;
    }
}

上述代码取消针对注册和学生数据的 ThenInclude 方法调用 。如果已选中课程,则突出显示的代码会检索:

  • 所选课程的 Enrollment 实体。
  • 每个 Enrollment 的 实体。

请注意,上述代码为 .AsNoTracking() 加上注释。对于跟踪的实体,仅可显式加载导航属性。

测试应用。对用户而言,该应用的行为与上一版本相同。

下一个教程将介绍如何更新相关数据。

其他资源Additional resources

下一页