Entity Framework 核心中的通用 where 条件
Generic where condition in Entity Framework Core
我正在构建一个使用工作单元设计模式的 Web API 应用程序。我创建了一个具有搜索功能的存储库:
public IQueryable<User> Search(SearchUserViewModel parameter)
{
// Find all user from database
var users = GetAll();
// Id has been defined.
if (parameter.Id != null)
users = users.Where(x => x.Id == parameter.Id.Value);
// Email has been defined.
if (parameter.Email != null && !string.IsNullOrWhiteSpace(parameter.Email.Text))
{
var emailSearch = parameter.Email;
switch (emailSearch.Mode)
{
case TextSearchMode.ContainIgnoreCase:
// TODO: Recheck.
users = users.Where(x => x.Email.Contains(emailSearch.Text));
break;
case TextSearchMode.Equal:
users = users.Where(x => x.Email.Equals(emailSearch.Text));
break;
case TextSearchMode.EqualIgnoreCase:
users = users.Where(x => x.Email.Equals(emailSearch.Text, StringComparison.OrdinalIgnoreCase));
break;
default:
users = users.Where(x => x.Email.Contains(emailSearch.Text));
break;
}
}
if (parameter.Password != null && !string.IsNullOrWhiteSpace(parameter.Password.Text))
{
var passwordSearch = parameter.Password;
switch (passwordSearch.Mode)
{
case TextSearchMode.ContainIgnoreCase:
// TODO: Recheck.
users = users.Where(x => x.Email.Contains(passwordSearch.Text));
break;
case TextSearchMode.Equal:
users = users.Where(x => x.Email.Equals(passwordSearch.Text));
break;
case TextSearchMode.EqualIgnoreCase:
users = users.Where(x => x.Email.Equals(passwordSearch.Text, StringComparison.OrdinalIgnoreCase));
break;
default:
users = users.Where(x => x.Email.Contains(passwordSearch.Text));
break;
}
}
return users;
}
枚举:
public class TextSearch
{
/// <summary>
/// Comparision mode.
/// </summary>
public TextSearchMode Mode { get; set; }
/// <summary>
/// Text which needs searching.
/// </summary>
public string Text { get; set; }
}
搜索功能目前运行良好。但我想减少我的代码。我想知道是否有一种方法可以创建这样的函数:
private IQueryable<T> SearchPropertyText(IQueryable<T> records, string propertyName, TextSearchMode mode)
{
switch (mode)
{
case Equal:
// TODO:
case EqualIgnoreCase:
// TODO:
default:
// TODO
}
return records;
}
然后我可以像这样使用函数:
users = SearchPropertyText(users, user.Email, emailSearch.Mode)
正如您在上面看到的我的代码,我在 TextSearchMode 枚举中列出了一些文本搜索模式,基于模式,我将搜索 属性 使用 Equal、EqualIgnoreCase、Contains、...
我正在考虑泛型函数,但我现在不知道如何创建它。
有人可以帮我吗?
而不是传递 user.Email,您需要传递定义如何从任何用户获得所需 属性 的 Func。此外,与其传递 TextSearchMode
,不如传递 TextSearch
对象,因为您还需要文本 属性 来执行搜索。这是一个代码示例:
private IQueryable<T> SearchPropertyText(IQueryable<T> records, Func<T, string> property, TextSearch search)
{
switch (search.Mode)
{
case TextSearchMode.ContainIgnoreCase:
records = records.Where(x => property(x).Contains(search.Text));
break;
case TextSearchMode.Equal:
records = records.Where(x => property(x).Equals(search.Text));
break;
case TextSearchMode.EqualIgnoreCase:
records = records.Where(x => property(x).Equals(search.Text, StringComparison.OrdinalIgnoreCase));
break;
default:
records = records.Where(x => property(x).Contains(search.Text));
break;
}
return records;
}
然后像这样使用它:
users = SearchPropertyText(users, user => user.Email, emailSearch);
我正在构建一个使用工作单元设计模式的 Web API 应用程序。我创建了一个具有搜索功能的存储库:
public IQueryable<User> Search(SearchUserViewModel parameter)
{
// Find all user from database
var users = GetAll();
// Id has been defined.
if (parameter.Id != null)
users = users.Where(x => x.Id == parameter.Id.Value);
// Email has been defined.
if (parameter.Email != null && !string.IsNullOrWhiteSpace(parameter.Email.Text))
{
var emailSearch = parameter.Email;
switch (emailSearch.Mode)
{
case TextSearchMode.ContainIgnoreCase:
// TODO: Recheck.
users = users.Where(x => x.Email.Contains(emailSearch.Text));
break;
case TextSearchMode.Equal:
users = users.Where(x => x.Email.Equals(emailSearch.Text));
break;
case TextSearchMode.EqualIgnoreCase:
users = users.Where(x => x.Email.Equals(emailSearch.Text, StringComparison.OrdinalIgnoreCase));
break;
default:
users = users.Where(x => x.Email.Contains(emailSearch.Text));
break;
}
}
if (parameter.Password != null && !string.IsNullOrWhiteSpace(parameter.Password.Text))
{
var passwordSearch = parameter.Password;
switch (passwordSearch.Mode)
{
case TextSearchMode.ContainIgnoreCase:
// TODO: Recheck.
users = users.Where(x => x.Email.Contains(passwordSearch.Text));
break;
case TextSearchMode.Equal:
users = users.Where(x => x.Email.Equals(passwordSearch.Text));
break;
case TextSearchMode.EqualIgnoreCase:
users = users.Where(x => x.Email.Equals(passwordSearch.Text, StringComparison.OrdinalIgnoreCase));
break;
default:
users = users.Where(x => x.Email.Contains(passwordSearch.Text));
break;
}
}
return users;
}
枚举:
public class TextSearch
{
/// <summary>
/// Comparision mode.
/// </summary>
public TextSearchMode Mode { get; set; }
/// <summary>
/// Text which needs searching.
/// </summary>
public string Text { get; set; }
}
搜索功能目前运行良好。但我想减少我的代码。我想知道是否有一种方法可以创建这样的函数:
private IQueryable<T> SearchPropertyText(IQueryable<T> records, string propertyName, TextSearchMode mode)
{
switch (mode)
{
case Equal:
// TODO:
case EqualIgnoreCase:
// TODO:
default:
// TODO
}
return records;
}
然后我可以像这样使用函数:
users = SearchPropertyText(users, user.Email, emailSearch.Mode)
正如您在上面看到的我的代码,我在 TextSearchMode 枚举中列出了一些文本搜索模式,基于模式,我将搜索 属性 使用 Equal、EqualIgnoreCase、Contains、...
我正在考虑泛型函数,但我现在不知道如何创建它。
有人可以帮我吗?
而不是传递 user.Email,您需要传递定义如何从任何用户获得所需 属性 的 Func。此外,与其传递 TextSearchMode
,不如传递 TextSearch
对象,因为您还需要文本 属性 来执行搜索。这是一个代码示例:
private IQueryable<T> SearchPropertyText(IQueryable<T> records, Func<T, string> property, TextSearch search)
{
switch (search.Mode)
{
case TextSearchMode.ContainIgnoreCase:
records = records.Where(x => property(x).Contains(search.Text));
break;
case TextSearchMode.Equal:
records = records.Where(x => property(x).Equals(search.Text));
break;
case TextSearchMode.EqualIgnoreCase:
records = records.Where(x => property(x).Equals(search.Text, StringComparison.OrdinalIgnoreCase));
break;
default:
records = records.Where(x => property(x).Contains(search.Text));
break;
}
return records;
}
然后像这样使用它:
users = SearchPropertyText(users, user => user.Email, emailSearch);