【问题标题】:Dynamic LINQ Like动态 LINQ 喜欢
【发布时间】:2009-10-31 15:43:10
【问题描述】:

如何为Like 子句编写动态LINQ 方法。

供参考,有Dynamic LINQ OrderBy on IEnumerable<T> / IQueryable<T>。我正在寻找一个类似的动态Like 子句。

我有以下类似的扩展方法:

public static IQueryable<T> Like<T>(this IQueryable<T> source, string propertyName, 
                                    string keyword)
{
    var type = typeof(T);
    var property = type.GetProperty(propertyName);
    var parameter = Expression.Parameter(type, "p");
    var propertyAccess = Expression.MakeMemberAccess(parameter, property);
    var constant = Expression.Constant("%" + keyword + "%");
    var methodExp = Expression.Call(
        null, 
        typeof(SqlMethods).GetMethod("Like", new[] { typeof(string), typeof(string) }),
        propertyAccess, 
        constant);
    var lambda = Expression.Lambda<Func<T, bool>>(methodExp, parameter);
    return source.Where(lambda);
}

上述方法报错

方法'Boolean Like(System.String, System.String)'不能在客户端使用;它仅用于转换为 SQL。

Dynamic LINQ OrderBy on IEnumerable<T> / IQueryable<T>修改的另一种方法:

public static IQueryable<T> ALike<T>(this IQueryable<T> source, string property, 
                                     string keyword)
{
    string[] props = property.Split('.');
    Type type = typeof(T);
    ParameterExpression arg = Expression.Parameter(type, "x");
    Expression expr = arg;

    foreach (string prop in props)
    {
        // use reflection (not ComponentModel) to mirror LINQ
        PropertyInfo pi = type.GetProperty(prop);
        expr = Expression.Property(expr, pi);
        type = pi.PropertyType;
    }
    var constant = Expression.Constant("%" + keyword + "%");
    var methodExp = Expression.Call(
        null, 
        typeof(SqlMethods).GetMethod("Like", new[] { typeof(string), typeof(string) }), 
        expr, 
        constant);
    Type delegateType = typeof(Func<,>).MakeGenericType(typeof(T), type);
    LambdaExpression lambda = Expression.Lambda(delegateType, methodExp, arg);
    object result = typeof(Queryable).GetMethods().Single(
            method => method.IsGenericMethodDefinition
                    && method.GetGenericArguments().Length == 2
                    && method.GetParameters().Length == 2)
            .MakeGenericMethod(typeof(T), type)
            .Invoke(null, new object[] { source, lambda });
    return (IQueryable<T>)result;
}

上述方法报错:

“System.Boolean”类型的表达式不能用于返回类型“System.String”

对此有什么想法吗?

【问题讨论】:

标签: linq dynamic


【解决方案1】:

类似:

static void Main() {
    using(var ctx= new DataClasses1DataContext()) {
        ctx.Log = Console.Out;
        var qry = ctx.Customers.WhereLike("CompanyName", "a%s");

        Console.WriteLine(qry.Count());
    }
}
static IQueryable<T> WhereLike<T>(this IQueryable<T> source,
        string propertyOrFieldName, string pattern) {
    var param = Expression.Parameter(typeof(T), "row");
    var body = Expression.Call(
        null,
        typeof(SqlMethods).GetMethod("Like",
            new[] { typeof(string), typeof(string) }),
        Expression.PropertyOrField(param, propertyOrFieldName),
        Expression.Constant(pattern, typeof(string)));
    var lambda = Expression.Lambda<Func<T, bool>>(body, param);
    return source.Where(lambda);
}
static IQueryable<T> WhereLike<T>(this IQueryable<T> source,
        string propertyOrFieldName, string pattern, char escapeCharacter) {
    var param = Expression.Parameter(typeof(T), "row");
    var body = Expression.Call(
        null,
        typeof(SqlMethods).GetMethod("Like",
            new[] { typeof(string), typeof(string), typeof(char) }),
        Expression.PropertyOrField(param, propertyOrFieldName),
        Expression.Constant(pattern, typeof(string)),
        Expression.Constant(escapeCharacter,typeof(char)));
    var lambda = Expression.Lambda<Func<T, bool>>(body, param);
    return source.Where(lambda);
}

你也可以考虑让它更可重用:

static void Main() {
    using(var ctx= new DataClasses1DataContext()) {
        ctx.Log = Console.Out;
        var qry1 = ctx.Customers.WhereInvoke<Customer, string>(
            "CompanyName", s => s.Contains("abc"));
        Console.WriteLine(qry1.Count());

        var qry2 = ctx.Customers.WhereInvoke<Customer, string>(
            "CompanyName", s => s.StartsWith("abc"));
        Console.WriteLine(qry2.Count());

        var qry3 = ctx.Customers.WhereInvoke<Customer, string>(
            "CompanyName", s => s.EndsWith("abc"));
        Console.WriteLine(qry3.Count());
    }
}
static IQueryable<TSource> WhereInvoke<TSource, TValue>(
        this IQueryable<TSource> source,
        string propertyOrFieldName,
        Expression<Func<TValue, bool>> func) {
    var param = Expression.Parameter(typeof(TSource), "row");
    var prop = Expression.PropertyOrField(param, propertyOrFieldName);
    if(prop.Type != typeof(TValue)) {
        throw new InvalidOperationException("The property must be " + typeof(TValue).Name);
    }
    var body = Expression.Invoke(func, prop);
    var lambda = Expression.Lambda<Func<TSource, bool>>(body, param);
    return source.Where(lambda);
}

【讨论】:

  • 上述方法也报错:无法在客户端使用Method 'Boolean Like(System.String, System.String)';它仅用于转换为 SQL。
  • 啊...你没有说清楚你的来源。除非您引用 VB 库,否则没有真正方便的“LIKE”等价物,这将对数据库提供程序起作用 - Contains/StartsWith/EndsWith 没有帮助吗?
  • 包含/开始/结束也给出错误。您为我一直在寻找的 Orderby 扩展提供了一个非常好的解决方案。该代码的美妙之处在于它还处理嵌套属性。我正在为“喜欢”尝试类似的...
  • @MarcGravell,如果我想搜索不是string 类型的属性,该怎么办?您的方法能否更通用,以便适用于所有类型?
  • @HeyJude WhereInvoke 中的代码根本没有提到 string。你有什么特别的想法吗?
【解决方案2】:

你知道SqlMethods.Like吗?

【讨论】:

  • 这给出了上面提到的相同错误:“方法 'Boolean Like(System.String, System.String)' 不能在客户端上使用;它仅用于转换为 SQL。”
  • @Ryan Elkins,你想要点赞的功能,但又不想调用数据库服务器?
【解决方案3】:

和你有同样的问题。 SqlMethods.Like 仅在 SQL 服务器上执行时有效,在内存集合上无效。因此,我创建了一个可以在集合上工作的 Like 评估器 - 请参阅我的博客文章 here

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2011-11-27
    • 2021-07-06
    • 1970-01-01
    • 1970-01-01
    • 2011-01-28
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多