首页 > 解决方案 > 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>也没有被翻译。

标签: sql-server.net-corelinq-to-sqlentity-framework-core

解决方案


如果您愿意使用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();

推荐阅读