EF6 - 将await关键字与Where()子句一起使用

时间:2014-12-10 12:11:15

标签: asynchronous asp.net-mvc-5 entity-framework-6 where async-await

我正在使用实体框架6编写MVC 5互联网应用程序,并在使用await子句时对使用.Where()关键字有疑问。

这是我的代码:

public async Task<Account> GetAccount(string userName)
{
    if (Session[userName] == null)
    {
        Account account = db.accounts.Where(a => a.userName.Equals(userName)).FirstOrDefault();
        if (account == null)
        {
            //log out
            return null;
        }
        Session[userName] = account;
    }
    return Session[userName] as Account;
}

我想在检索Account object时使用await关键字,如下所示:

Account account = await db.accounts.Where(a => a.userName.Equals(userName)).FirstOrDefault();

使用await子句时可以使用.Where()关键字吗?

提前致谢。

2 个答案:

答案 0 :(得分:12)

EF提供了WhereAsync()方法(显然因为它可能会阻止,因为LINQ使用延迟执行),但是因为你正在执行FirstOrDefault() 你可以简单地使用FirstOrDefaultAsync()方法:

Account account = await db.accounts.FirstOrDefaultAsync(a => a.userName.Equals(userName));

请参阅MSDN

答案 1 :(得分:5)

await关键字只能用于返回&#34;任务...&#34;的方法,既不是.Where也不是.FirstOrDefault(这是链中的最后一个方法,以及因此,await关键字将应用于)返回Task<IEnumerable<Account>>

从理论上讲,你可以编写自己的扩展方法,它只包含.Where.FirstOrDefault方法。

此外,这个问题并不完全符合EF的要求,而是纯粹的&#39; C#问题。

public static class ExtensionMethods
{
    public static async Task<IEnumerable<T>> WhereAsync<T>(this IEnumerable<T> source, Func<T, bool> selector)
    {
        return await Task.Run(() => source.Where(selector));
    }
}

虽然那会有点矫枉过正。

您可以将整个方法包装在一个Task中,因此您的最终代码将类似于:

public async Task<Account> GetAccount(string userName)
{
    return await Task.Run(() =>
    {
        if (Session[userName] == null)
        {
            Account account = db.accounts.Where(a => a.userName.Equals(userName)).FirstOrDefault();
            if (account == null)
            {
                //log out
                return null;
            }
            Session[userName] = account;
        }
        return Session[userName] as Account;
    });
}