实体框架 - 渴望加载相关实体的相关实体

时间:2013-05-10 19:16:43

标签: visual-studio-2012 entity-framework-5 sql-server-2012-express

我正在使用实体框架5在C#(.NET 4.5)中编写一个简单的数据库应用程序。我可能需要或不需要加载单个实体的相关实体。如果碰巧我需要加载实体的相关实体,我想急切加载相关实体的相关实体。基本上,我试图避免“SELECT N + 1”问题。希望以下代码能够成为我想要做的事情:

using (var ctx = new DbContext())
{
    // Find a single instance of a person.  I don't want to eagerly load
    // the person's friends at this point because I may not need this
    // information.
    var person = ctx.Persons.Single(x => x.PersonID == 12);

    // Many lines of code...
    // Many lines of code...
    // Many lines of code...

    // Okay, if we have reached this point in the code, the person wants to 
    // send all his friends a postcard, so we need the person's friends and
    // their addresses.

    // I want to eagerly load the person's friends' addresses, but the Include
    // method is not allowed.  The property "Friends" is just an ObservableCollection.
    var friends = person.Friends.Include("Address").ToList();

    // So, I must do the following:
    var friends = person.Friends.ToList();

    // Now I will output the address of each friend. This is where I have the 
    // SELECT N+1 problem.
    foreach(var friend in friends)
    {
        // Every time this line is executed a query (SELECT statement) is sent
        // to the database.
        Console.WriteLine(friend.Address.Street);

    }

}

关于我应该做什么的任何想法?

1 个答案:

答案 0 :(得分:1)

对于显式加载这是一个很好的情况 - 除了渴望和延迟加载之外,第三个使用Entity Framework加载相关实体的选项:

using (var ctx = new DbContext())
{
    var person = ctx.Persons.Single(x => x.PersonID == 12);

    // ...

    // the following issues one single DB query
    person.Friends = ctx.Entry(person).Collection(p => p.Friends).Query()
        .Include(f => f.Address) // = .Include("Address")
        .ToList();

    foreach(var friend in person.Friends)
    {
        // No DB query because all friends including addresses
        // have already been loaded
        Console.WriteLine(friend.Address.Street);
    }
}

此处的关键是.Query(),它返回Friends集合的可查询对象,允许您为好友集合添加任意其他查询逻辑 - 如过滤,排序,加入其他相关数据(= Include),聚合(例如朋友Count)等等。