AspNetCore.Docs/aspnetcore/data/ef-mvc/read-related-data.md

268 lines
18 KiB
Markdown
Raw Normal View History

2016-10-29 01:35:15 +08:00
---
title: "Tutorial: Read related data - ASP.NET MVC with EF Core"
description: "In this tutorial you'll read and display related data -- that is, data that the Entity Framework loads into navigation properties."
author: tdykstra
2019-08-22 23:46:35 +08:00
ms.author: riande
2019-09-28 19:52:34 +08:00
ms.date: 09/28/2019
ms.topic: tutorial
2016-10-29 01:35:15 +08:00
uid: data/ef-mvc/read-related-data
---
2016-12-08 07:35:57 +08:00
# Tutorial: Read related data - ASP.NET MVC with EF Core
2016-10-29 01:35:15 +08:00
2017-08-18 02:36:07 +08:00
In the previous tutorial, you completed the School data model. In this tutorial, you'll read and display related data -- that is, data that the Entity Framework loads into navigation properties.
2016-10-29 01:35:15 +08:00
The following illustrations show the pages that you'll work with.
![Courses Index page](read-related-data/_static/courses-index.png)
![Instructors Index page](read-related-data/_static/instructors-index.png)
In this tutorial, you:
> [!div class="checklist"]
> * Learn how to load related data
> * Create a Courses page
> * Create an Instructors page
> * Learn about explicit loading
## Prerequisites
2019-03-28 05:47:23 +08:00
* [Create a complex data model](complex-data-model.md)
## Learn how to load related data
2016-10-29 01:35:15 +08:00
There are several ways that Object-Relational Mapping (ORM) software such as Entity Framework can load related data into the navigation properties of an entity:
* [Eager loading:](/ef/core/querying/related-data) When the entity is read, related data is retrieved along with it. This typically results in a single join query that retrieves all of the data that's needed. You specify eager loading in Entity Framework Core by using the `Include` and `ThenInclude` methods.
2016-10-29 01:35:15 +08:00
![Eager loading example](read-related-data/_static/eager-loading.png)
You can retrieve some of the data in separate queries, and EF "fixes up" the navigation properties. That is, EF automatically adds the separately retrieved entities where they belong in navigation properties of previously retrieved entities. For the query that retrieves related data, you can use the `Load` method instead of a method that returns a list or object, such as `ToList` or `Single`.
![Separate queries example](read-related-data/_static/separate-queries.png)
* [Explicit loading:](/ef/core/querying/related-data) When the entity is first read, related data isn't retrieved. You write code that retrieves the related data if it's needed. As in the case of eager loading with separate queries, explicit loading results in multiple queries sent to the database. The difference is that with explicit loading, the code specifies the navigation properties to be loaded. In Entity Framework Core 1.1 you can use the `Load` method to do explicit loading. For example:
![Explicit loading example](read-related-data/_static/explicit-loading.png)
2016-10-29 01:35:15 +08:00
* [Lazy loading:](/ef/core/querying/related-data) When the entity is first read, related data isn't retrieved. However, the first time you attempt to access a navigation property, the data required for that navigation property is automatically retrieved. A query is sent to the database each time you try to get data from a navigation property for the first time. Entity Framework Core 1.0 doesn't support lazy loading.
2016-10-29 01:35:15 +08:00
### Performance considerations
If you know you need related data for every entity retrieved, eager loading often offers the best performance, because a single query sent to the database is typically more efficient than separate queries for each entity retrieved. For example, suppose that each department has ten related courses. Eager loading of all related data would result in just a single (join) query and a single round trip to the database. A separate query for courses for each department would result in eleven round trips to the database. The extra round trips to the database are especially detrimental to performance when latency is high.
On the other hand, in some scenarios separate queries is more efficient. Eager loading of all related data in one query might cause a very complex join to be generated, which SQL Server can't process efficiently. Or if you need to access an entity's navigation properties only for a subset of a set of the entities you're processing, separate queries might perform better because eager loading of everything up front would retrieve more data than you need. If performance is critical, it's best to test performance both ways in order to make the best choice.
## Create a Courses page
2016-10-29 01:35:15 +08:00
The `Course` entity includes a navigation property that contains the `Department` entity of the department that the course is assigned to. To display the name of the assigned department in a list of courses, you need to get the `Name` property from the `Department` entity that's in the `Course.Department` navigation property.
2016-10-29 01:35:15 +08:00
Create a controller named `CoursesController` for the `Course` entity type, using the same options for the **MVC Controller with views, using Entity Framework** scaffolder that you did earlier for the `StudentsController`, as shown in the following illustration:
2016-10-29 01:35:15 +08:00
![Add Courses controller](read-related-data/_static/add-courses-controller.png)
Open `CoursesController.cs` and examine the `Index` method. The automatic scaffolding has specified eager loading for the `Department` navigation property by using the `Include` method.
2016-10-29 01:35:15 +08:00
Replace the `Index` method with the following code that uses a more appropriate name for the `IQueryable` that returns Course entities (`courses` instead of `schoolContext`):
[!code-csharp[](intro/samples/cu/Controllers/CoursesController.cs?name=snippet_RevisedIndexMethod)]
2016-10-29 01:35:15 +08:00
Open `Views/Courses/Index.cshtml` and replace the template code with the following code. The changes are highlighted:
2016-10-29 01:35:15 +08:00
2020-07-10 06:22:19 +08:00
[!code-cshtml[](intro/samples/cu/Views/Courses/Index.cshtml?highlight=4,7,15-17,34-36,44)]
2016-10-29 01:35:15 +08:00
You've made the following changes to the scaffolded code:
* Changed the heading from **Index** to **Courses**.
2016-10-29 01:35:15 +08:00
* Added a **Number** column that shows the `CourseID` property value. By default, primary keys aren't scaffolded because normally they're meaningless to end users. However, in this case the primary key is meaningful and you want to show it.
2016-10-29 01:35:15 +08:00
* Changed the **Department** column to display the department name. The code displays the `Name` property of the `Department` entity that's loaded into the `Department` navigation property:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```html
2016-10-29 01:35:15 +08:00
@Html.DisplayFor(modelItem => item.Department.Name)
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
2017-09-20 18:36:07 +08:00
Run the app and select the **Courses** tab to see the list with department names.
2016-10-29 01:35:15 +08:00
![Courses Index page](read-related-data/_static/courses-index.png)
## Create an Instructors page
2016-10-29 01:35:15 +08:00
In this section, you'll create a controller and view for the `Instructor` entity in order to display the Instructors page:
2016-10-29 01:35:15 +08:00
![Instructors Index page](read-related-data/_static/instructors-index.png)
This page reads and displays related data in the following ways:
* The list of instructors displays related data from the `OfficeAssignment` entity. The `Instructor` and `OfficeAssignment` entities are in a one-to-zero-or-one relationship. You'll use eager loading for the `OfficeAssignment` entities. As explained earlier, eager loading is typically more efficient when you need the related data for all retrieved rows of the primary table. In this case, you want to display office assignments for all displayed instructors.
2016-10-29 01:35:15 +08:00
* When the user selects an instructor, related `Course` entities are displayed. The `Instructor` and `Course` entities are in a many-to-many relationship. You'll use eager loading for the `Course` entities and their related `Department` entities. In this case, separate queries might be more efficient because you need courses only for the selected instructor. However, this example shows how to use eager loading for navigation properties within entities that are themselves in navigation properties.
2016-10-29 01:35:15 +08:00
* When the user selects a course, related data from the `Enrollments` entity set is displayed. The `Course` and `Enrollment` entities are in a one-to-many relationship. You'll use separate queries for `Enrollment` entities and their related `Student` entities.
2016-10-29 01:35:15 +08:00
### Create a view model for the Instructor Index view
The Instructors page shows data from three different tables. Therefore, you'll create a view model that includes three properties, each holding the data for one of the tables.
In the *SchoolViewModels* folder, create `InstructorIndexData.cs` and replace the existing code with the following code:
2016-10-29 01:35:15 +08:00
[!code-csharp[](intro/samples/cu/Models/SchoolViewModels/InstructorIndexData.cs)]
2016-10-29 01:35:15 +08:00
### Create the Instructor controller and views
Create an Instructors controller with EF read/write actions as shown in the following illustration:
![Add Instructors controller](read-related-data/_static/add-instructors-controller.png)
Open `InstructorsController.cs` and add a using statement for the ViewModels namespace:
2016-10-29 01:35:15 +08:00
[!code-csharp[](intro/samples/cu/Controllers/InstructorsController.cs?name=snippet_Using)]
2016-10-29 01:35:15 +08:00
Replace the Index method with the following code to do eager loading of related data and put it in the view model.
[!code-csharp[](intro/samples/cu/Controllers/InstructorsController.cs?name=snippet_EagerLoading)]
2016-10-29 01:35:15 +08:00
The method accepts optional route data (`id`) and a query string parameter (`courseID`) that provide the ID values of the selected instructor and selected course. The parameters are provided by the **Select** hyperlinks on the page.
2017-03-21 05:05:56 +08:00
The code begins by creating an instance of the view model and putting in it the list of instructors. The code specifies eager loading for the `Instructor.OfficeAssignment` and the `Instructor.CourseAssignments` navigation properties. Within the `CourseAssignments` property, the `Course` property is loaded, and within that, the `Enrollments` and `Department` properties are loaded, and within each `Enrollment` entity the `Student` property is loaded.
2016-10-29 01:35:15 +08:00
[!code-csharp[](intro/samples/cu/Controllers/InstructorsController.cs?name=snippet_ThenInclude)]
2016-10-29 01:35:15 +08:00
Since the view always requires the `OfficeAssignment` entity, it's more efficient to fetch that in the same query. Course entities are required when an instructor is selected in the web page, so a single query is better than multiple queries only if the page is displayed more often with a course selected than without.
2016-10-29 01:35:15 +08:00
2017-03-21 05:05:56 +08:00
The code repeats `CourseAssignments` and `Course` because you need two properties from `Course`. The first string of `ThenInclude` calls gets `CourseAssignment.Course`, `Course.Enrollments`, and `Enrollment.Student`.
You can read more about including multiple levels of related data [here.](/ef/core/querying/related-data/eager#including-multiple-levels)
[!code-csharp[](intro/samples/cu/Controllers/InstructorsController.cs?name=snippet_ThenInclude&highlight=3-6)]
2017-03-21 05:05:56 +08:00
At that point in the code, another `ThenInclude` would be for navigation properties of `Student`, which you don't need. But calling `Include` starts over with `Instructor` properties, so you have to go through the chain again, this time specifying `Course.Department` instead of `Course.Enrollments`.
[!code-csharp[](intro/samples/cu/Controllers/InstructorsController.cs?name=snippet_ThenInclude&highlight=7-9)]
2016-10-29 01:35:15 +08:00
The following code executes when an instructor was selected. The selected instructor is retrieved from the list of instructors in the view model. The view model's `Courses` property is then loaded with the `Course` entities from that instructor's `CourseAssignments` navigation property.
2016-10-29 01:35:15 +08:00
[!code-csharp[](intro/samples/cu/Controllers/InstructorsController.cs?range=56-62)]
2016-10-29 01:35:15 +08:00
The `Where` method returns a collection, but in this case the criteria passed to that method result in only a single Instructor entity being returned. The `Single` method converts the collection into a single `Instructor` entity, which gives you access to that entity's `CourseAssignments` property. The `CourseAssignments` property contains `CourseAssignment` entities, from which you want only the related `Course` entities.
2016-10-29 01:35:15 +08:00
You use the `Single` method on a collection when you know the collection will have only one item. The `Single` method throws an exception if the collection passed to it's empty or if there's more than one item. An alternative is `SingleOrDefault`, which returns a default value (null in this case) if the collection is empty. However, in this case that would still result in an exception (from trying to find a `Courses` property on a null reference), and the exception message would less clearly indicate the cause of the problem. When you call the `Single` method, you can also pass in the Where condition instead of calling the `Where` method separately:
2016-10-29 01:35:15 +08:00
2016-11-18 13:03:07 +08:00
```csharp
2016-10-29 01:35:15 +08:00
.Single(i => i.ID == id.Value)
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
Instead of:
2016-11-18 13:03:07 +08:00
```csharp
.Where(i => i.ID == id.Value).Single()
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
Next, if a course was selected, the selected course is retrieved from the list of courses in the view model. Then the view model's `Enrollments` property is loaded with the Enrollment entities from that course's `Enrollments` navigation property.
[!code-csharp[](intro/samples/cu/Controllers/InstructorsController.cs?range=64-69)]
2016-10-29 01:35:15 +08:00
### Tracking vs no-tracking
No-tracking queries are useful when the results are used in a read-only scenario. They're generally quicker to execute because there's no need to set up the change tracking information. If the entities retrieved from the database don't need to be updated, then a no-tracking query is likely to perform better than a tracking query.
In some cases a tracking query is more efficient than a no-tracking query. For more information, see [Tracking vs. No-Tracking Queries](/ef/core/querying/tracking).
2016-10-29 01:35:15 +08:00
### Modify the Instructor Index view
In `Views/Instructors/Index.cshtml`, replace the template code with the following code. The changes are highlighted.
2016-10-29 01:35:15 +08:00
:::moniker range=">= aspnetcore-2.2"
[!code-cshtml[](intro/samples/5cu-snap/Views/Instructors/Index.cshtml?range=1-62&highlight=1,3-7,15-19,24,26-31,41-52,54)]
:::moniker-end
:::moniker range="<= aspnetcore-2.1"
[!code-cshtml[](intro/samples/cu/Views/Instructors/Index1.cshtml?range=1-62&highlight=1,3-7,15-19,24,26-31,41-52,54)]
:::moniker-end
2016-10-29 01:35:15 +08:00
You've made the following changes to the existing code:
* Changed the model class to `InstructorIndexData`.
* Changed the page title from **Index** to **Instructors**.
* Added an **Office** column that displays `item.OfficeAssignment.Location` only if `item.OfficeAssignment` isn't null. (Because this is a one-to-zero-or-one relationship, there might not be a related OfficeAssignment entity.)
2016-10-29 01:35:15 +08:00
```cshtml
2016-10-29 01:35:15 +08:00
@if (item.OfficeAssignment != null)
{
@item.OfficeAssignment.Location
}
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
2019-09-28 19:52:34 +08:00
* Added a **Courses** column that displays courses taught by each instructor. For more information, see the [Explicit line transition](xref:mvc/views/razor#explicit-line-transition) section of the Razor syntax article.
2016-10-29 01:35:15 +08:00
* Added code that conditionally adds a Bootstrap CSS class to the `tr` element of the selected instructor. This class sets a background color for the selected row.
2016-10-29 01:35:15 +08:00
* Added a new hyperlink labeled **Select** immediately before the other links in each row, which causes the selected instructor's ID to be sent to the `Index` method.
```cshtml
2016-10-29 01:35:15 +08:00
<a asp-action="Index" asp-route-id="@item.ID">Select</a> |
2016-11-18 13:03:07 +08:00
```
2016-10-29 01:35:15 +08:00
2017-09-20 18:36:07 +08:00
Run the app and select the **Instructors** tab. The page displays the Location property of related OfficeAssignment entities and an empty table cell when there's no related OfficeAssignment entity.
2016-10-29 01:35:15 +08:00
![Instructors Index page nothing selected](read-related-data/_static/instructors-index-no-selection.png)
In the `Views/Instructors/Index.cshtml` file, after the closing table element (at the end of the file), add the following code. This code displays a list of courses related to an instructor when an instructor is selected.
2016-10-29 01:35:15 +08:00
2021-04-26 04:09:41 +08:00
[!code-cshtml[](intro/samples/cu/Views/Instructors/Index1.cshtml?range=63-99)]
2016-10-29 01:35:15 +08:00
This code reads the `Courses` property of the view model to display a list of courses. It also provides a **Select** hyperlink that sends the ID of the selected course to the `Index` action method.
2017-09-20 18:36:07 +08:00
Refresh the page and select an instructor. Now you see a grid that displays courses assigned to the selected instructor, and for each course you see the name of the assigned department.
2016-10-29 01:35:15 +08:00
![Instructors Index page instructor selected](read-related-data/_static/instructors-index-instructor-selected.png)
After the code block you just added, add the following code. This displays a list of the students who are enrolled in a course when that course is selected.
[!code-cshtml[](intro/samples/cu/Views/Instructors/Index1.cshtml?range=101-125)]
2016-10-29 01:35:15 +08:00
This code reads the `Enrollments` property of the view model in order to display a list of students enrolled in the course.
2016-10-29 01:35:15 +08:00
2017-09-20 18:36:07 +08:00
Refresh the page again and select an instructor. Then select a course to see the list of enrolled students and their grades.
2016-10-29 01:35:15 +08:00
![Instructors Index page instructor and course selected](read-related-data/_static/instructors-index.png)
## About explicit loading
2016-10-29 01:35:15 +08:00
When you retrieved the list of instructors in `InstructorsController.cs`, you specified eager loading for the `CourseAssignments` navigation property.
2016-10-29 01:35:15 +08:00
Suppose you expected users to only rarely want to see enrollments in a selected instructor and course. In that case, you might want to load the enrollment data only if it's requested. To see an example of how to do explicit loading, replace the `Index` method with the following code, which removes eager loading for `Enrollments` and loads that property explicitly. The code changes are highlighted.
2016-10-29 01:35:15 +08:00
[!code-csharp[](intro/samples/cu/Controllers/InstructorsController.cs?name=snippet_ExplicitLoading&highlight=23-29)]
2016-10-29 01:35:15 +08:00
The new code drops the `ThenInclude` method calls for enrollment data from the code that retrieves instructor entities. It also drops `AsNoTracking`. If an instructor and course are selected, the highlighted code retrieves `Enrollment` entities for the selected course, and `Student` entities for each `Enrollment`.
2016-10-29 01:35:15 +08:00
2017-09-20 18:36:07 +08:00
Run the app, go to the Instructors Index page now and you'll see no difference in what's displayed on the page, although you've changed how the data is retrieved.
2016-10-29 01:35:15 +08:00
## Get the code
2021-03-10 12:40:27 +08:00
[Download or view the completed application.](https://github.com/dotnet/AspNetCore.Docs/tree/main/aspnetcore/data/ef-mvc/intro/samples/cu-final)
## Next steps
2016-10-29 01:35:15 +08:00
In this tutorial, you:
2016-10-29 01:35:15 +08:00
> [!div class="checklist"]
> * Learned how to load related data
> * Created a Courses page
> * Created an Instructors page
> * Learned about explicit loading
2019-03-28 05:47:23 +08:00
Advance to the next tutorial to learn how to update related data.
> [!div class="nextstepaction"]
> [Update related data](update-related-data.md)