In this article, we are going to learn about Filtered Include in Entity Framework Core, what operations it supports, and what are its use cases.

To download the source code for this article, you can visit our GitHub repository.

Let’s start.

Why Do We Need Filtered Include

When we use eager loading in EF Core, we use the Include method to load related entities.

Support Code Maze on Patreon to get rid of ads and get the best discounts on our products!
Become a patron at Patreon!

But what if we want to filter or sort these related entities?

We can do that with a new feature in EF Core 5.0 that allows using LINQ methods inside the Include method. We can use this feature on the collection navigation property, not the reference navigation property.

To start with an example, let’s create two entities: Course and Student, with a One-To-Many relationship.

First the Student entity:

public class Student
{
    public int Id { get; set; }
    public string? Name { get; set; }
    public int Mark { get; set; }

    public int CourseId { get; set; }
    public Course? Course { get; set; }
}

And theCourse entity:

public class Course
{
    public int Id { get; set; }
    public string? Title { get; set; }

    public ICollection<Student>? Students { get; set; }
}

Note: We create this relationship for the sake of simplicity and will use it in all the following examples in our article.

Let’s see an example of how Filtered Include affects the result:

var query = context.Courses.Include(x => x.Students.Where(x => x.Mark > 90)).ToList();

The query returns all the courses with the students that have a mark greater than 90.

Supported Operations

The supported operations inside Include are: Where, OrderBy, OrderByDescending, ThenBy, ThenByDescending, Skip, and Take. 

So the following query will work:

var goodQuery = context.Courses.Include(c => c.Students.Where(s => s.Mark > 50)).ToList();

But the next query will not:

var badQuery = context.Courses.Include(c => c.Students.Any(s => s.Mark > 50)).ToList();

This query throws InvalidOperationException because the Any method is not supported.

Stand-Alone Filter

The applied filter on Include must be stand-alone, i.e. it must work independently of Include.

To make it clear, let’s see an example:

var goodQuery = context.Courses.Include(c => c.Students.Where(s => s.Id == s.Course.Id)).ToList();

This query is correct and works because Where(s => s.Id == s.Course.Id) can work independently:

var query = context.Students.Where(s => s.Id == s.Course.Id).ToList();

But if we rewrite the query:

var badQuery = context.Courses.Include(c => c.Students.Where(s => s.Id == c.Id)).ToList();

This query throws InvalidOperationException because the LINQ expression ‘c’ could not be translated.

Filtering on Multiple Include

We can have only one filter per collection navigation. So when we need to include the same navigation multiple times, we should apply the same filter:

var goodQuery = context.Courses
    .Include(c => c.Students.Where(s => s.Mark > 50)).ThenInclude(s => s.Entity1)
    .Include(c => c.Students.Where(s => s.Mark > 50)).ThenInclude(s => s.Entity2)
    .ToList();

But if we try to apply different filters for the same navigation:

var badQuery = context.Courses
    .Include(c => c.Students.Where(s => s.Mark > 50)).ThenInclude(s => s.Entity1)
    .Include(c => c.Students.Where(s => s.Mark <= 50)).ThenInclude(s => s.Entity2)
    .ToList();

It throws InvalidOperationException.

Filtered Include with Tracking Queries

In the case of tracking queries, when we execute a query, the relevant entities will be stored in the change tracker. Because of this, we may get unexpected results.

So let’s see an example:

var query1 = context.Courses
    .Include(c => c.Students.Where(s => s.Mark > 50)).ToList(); 
var query2 = context.Courses
    .Include(c => c.Students.Where(s => s.Mark <= 50)).ToList();

After executing the queries, the result is an aggregate between the first and second predicates, so we return all the students related to courses. To solve this issue we can use a new context or AsNoTracking method:

var query1 = context.Courses
    .AsNoTracking()
    .Include(c => c.Students.Where(s => s.Mark > 50)).ToList();
var query2 = context.Courses
    .AsNoTracking()
    .Include(c => c.Students.Where(s => s.Mark <= 50)).ToList();

Another example to show the effect:

var courses = context.Courses.Include(c => c.Students.Where(s => s.Mark > 50)).ToList();
var students = context.Students!.Where(s => s.Mark <= 50).ToList();

The courses contains all the students as in the previous example.

Filtered Include and Select Method

First, let’s explain the relationship between Select and Include. When we use Select, we don’t need Include:

var query1 = context.Courses
    .Select(c => new
    {
        c.Id,
        c.Title
    }).ToList();

var query2 = context.Courses.Include(c => c.Students)
    .Select(c => new
    {
        c.Id,
        c.Title
    }).ToList();

Both query1 and query2 translate to SQL in the same way. Due to the projection operation with the Select method in the second query, Include will be ignored.

We can have a look at another example:

var query3 = context.Courses //.Include(x => x.Students)
    .Select(c => new
    {
        c.Id,
        c.Title,
        Students = c.Students.ToList()
    }).ToList();

Now, since Select uses Students collection, EF Core does a left-join whether or not we use Include.

Back to Filtered Include:

var query4 = context.Courses
    .Include(x => x.Students.Where(x => x.Mark > 50))
    .Select(c => new Course 
    { 
        Id = c.Id,
        Title = c.Title,
        Students = c.Students!.ToList() 
    }).ToList();

In this case EF Core ignores Include. The result contains all the students and not only those with marks greater than 50.

If we want to use Select and filter the collection navigation at the same time, how can we do this?

We can do this by filtering inside Select:

var query5 = context.Courses
    .Select(c => new Course 
    { 
        Id = c.Id,
        Title = c.Title,
        Students = c.Students.Where(x => x.Mark > 50).ToList()
    }).ToList();

Conclusion

In this article, we’ve learned about Filtered Include and its implementation, and we have seen many uses that lead to exceptions and which should be avoided.

Finally, we have tried to explain the complex relationship between Include and Select methods with many examples.

Liked it? Take a second to support Code Maze on Patreon and get the ad free reading experience!
Become a patron at Patreon!