1

我正在使用 EF5 有一个由字符串、日期、可为空的日期和整数组成的 20 个大型属性实体。是否可以使用 Linq 对每个字段进行包含。

我不想为每个字段都构建一个 linq 语句

results = list.Where(house => house.Date1.ToString().Contains(search)||
                              house.Address1.Contains(search)||
                              house.Address2.Contains(search)||
                              house.Address3.Contains(search)||
                              .........................)

我想我想将所有属性转换为字符串(在必要时),然后对它们中的每一个进行包含?理想情况下,这将发生在数据库级别。

我想我不能将它们连接起来,因为如果两个字段的结尾和开头创建了搜索的文本,它可能会导致错误匹配?有任何想法吗?

更新

我目前是使用 LinqKit AsExpandable 和 EF 转换函数到基于类型的字符串的多个字段的 LINQ string[]的第二个答案。但是我似乎无法找到将 DateTime 转换为字符串的 SqlFunction

4

2 回答 2

5

所以这里的总体思路是使用PredicateBuilder对一组表达式进行 OR 组合,每个表达式代表Contains对给定属性的检查。您可以使用反射来查找给定类型的所有属性以创建表达式。

public static Expression<Func<T, bool>> SearchAllFields<T>(string searchText)
{
    var t = Expression.Parameter(typeof(T));
    Expression body = Expression.Constant(false);

    var containsMethod = typeof(string).GetMethod("Contains"
        , new[] { typeof(string) });
    var toStringMethod = typeof(object).GetMethod("ToString");

    var stringProperties = typeof(T).GetProperties()
        .Where(property => property.PropertyType == typeof(string));

    foreach (var property in stringProperties)
    {
        var stringValue = Expression.Call(Expression.Property(t, property.Name),
            toStringMethod);
        var nextExpression = Expression.Call(stringValue,
            containsMethod,
            Expression.Constant(searchText));

        body = Expression.OrElse(body, nextExpression);
    }

    return Expression.Lambda<Func<T, bool>>(body, t);
}
于 2013-09-11T15:07:51.490 回答
1

我最近创建了一个 nuget 包,它使用扩展方法解决了这个问题

https://www.nuget.org/packages/NinjaNye.SearchExtensions/

这将允许以下代码...

using NinjaNye.SearchExtensions;
//...
var result = list.Search(h => search, h.Date1.ToString(),
                                      h.Address1,
                                      h.Address2,
                                      h.Address3,
                                      ...etc...)

如果您想使用代码,请查看 github 页面: https ://github.com/ninjanye/SearchExtensions

当连接到 sql 数据库时,您会得到类似于以下 sql 的内容:

SELECT [Extent1].[Date1] AS [Date1], 
       [Extent1].[Address1] AS [Address1], 
       [Extent1].[Address2] AS [Address2], 
       [Extent1].[Address3] AS [Address3] 
FROM   [dbo].[Table] AS [Extent1]
WHERE ([Extent1].[Address1] LIKE N'%searchTerm%') 
   OR ([Extent1].[Address2] LIKE N'%searchTerm%') 
   OR ([Extent1].[Address3] LIKE N'%searchTerm%')

希望这可以帮助

于 2013-12-12T12:27:59.473 回答