sql-server - EF Core - 无法查询 entity.Name 是否存在于列表中
问题描述
使用Entity Framework Core 3.1.7
我在包含产品的数据库中有一个表。
public class Product
{
public int Id {get; set;}
public string Name {get; set;}
}
然后,我希望用户能够使用搜索字段在 UI 表中查找某些产品。当查询进来时,我尝试执行以下操作:
var searchParameters = query.SearchParameters.ToLower().Split(' ', ',', '+').Distinct();
var result = _context.Products
.Where(p => searchParameters.Any()
&& (searchParameters.Any(x => p.Name.ToLower().Contains(x)) //Version 1
).ToList();
或替代方案
searchParameters.Any(x => EF.Functions.Like(p.Name, "%" + x + "%")) //Version 2
但是,我调整了我得到的这个看似简单的东西:
无法翻译 LINQ 表达式 'DbSet .Where(p => __searchParameters_0 .Any(x => p.Name.ToLower().Contains(x)))'。以可翻译的形式重写查询,或通过插入对 AsEnumerable()、AsAsyncEnumerable()、ToList() 或 ToListAsync() 的调用显式切换到客户端评估
我意识到这.ToLower()
将是一个问题,所以我想为不区分大小写的搜索运行一个 LIKE 语句,就像 SQL 查询一样。但即便如此,List<string>
也没有被翻译。
解决方案
如果您愿意使用LINQKit(或模拟谓词构建器部分),则可以使用扩展方法将Any(
...Contains)
表达式扩展为“或”表达式:
public static class LinqKitExt { // using LINQKit
// keyFne - extract string key from row
// searchTerms - IEnumerable<string> where one must be contained by a row's key
// dbq.Where(r => searchTerms.Any(s => keyFne(r).Contains(s)))
public static IQueryable<T> WhereContainsAny<T>(this IQueryable<T> dbq, Expression<Func<T,string>> keyFne, IEnumerable<string> searchTerms) {
var pred = PredicateBuilder.New<T>();
foreach (var s in searchTerms)
pred = pred.Or(r => keyFne.Invoke(r).Contains(s));
return dbq.Where((Expression<Func<T,bool>>)pred.Expand());
}
}
(以及 Where/OrderBy[Descending] Any/All Contains/StartsWith 的 51 个其他变体。)
然后你可以像这样使用它
var result = _context.Products
.WhereContainsAny(r => r.Name, searchParameters)
.ToList();
PS追求另一个答案,我意识到将测试拉给调用者消除了大多数变化:
// searchTerms - IEnumerable<TKey> where all must be in a row's key
// testFne(row,searchTerm) - test one of searchTerms against a row
// dbq.Where(r => searchTerms.Any(s => testFne(r,s)))
public static IQueryable<T> WhereAny<T,TKey>(this IQueryable<T> dbq, IEnumerable<TKey> searchTerms, Expression<Func<T, TKey, bool>> testFne) {
var pred = PredicateBuilder.New<T>();
foreach (var s in searchTerms)
pred = pred.Or(r => testFne.Invoke(r, s));
return dbq.Where((Expression<Func<T,bool>>)pred.Expand());
}
然后你只需调用:
var result = _context.Products
.WhereAny(searchParameters, (r,s) => r.Name.Contains(s))
.ToList();
推荐阅读
- shell - if (( ... )) 的 POSIX 兼容替代方案是什么?
- mongodb - 我如何在 kubernetes 中使用 mongoconnector
- typescript - 函数返回类型取决于是否传递了可选参数
- python-3.x - 在不加载内存的情况下读取 Shapefile 的 Zip 时出现问题
- scala - 将通用附加函数迁移到 Scala 2.13 集合
- java - 如何使用嵌套的 if 语句而不是 System.exit(0); 阻止程序在蓝图类中运行?
- python - 从数据框中检索许多聚合计数的最佳方法?
- assembly - nasm 指令和宏的缩进
- solr - Solr 空间字段上的过滤器查询返回所有具有非空值的文档
- java - 如何将我的数据插入二维多维数组?