我正在使用实体框架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()
关键字吗?
提前致谢。
答案 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;
});
}