4

我正在尝试在允许用户创建涉及列、操作和值的过滤条件的网格中实现搜索/过滤。

示例:Column1 包含“somevalue”

当所选列包含字符串类型时,以下代码可以正常工作:

case WhereOperation.Contains:
    Expression condition = Expression.Call(
       memberAccess, //memberAccess is a MemberExpression of the property bound to the column.
       typeof(string).GetMethod("Contains"),
       Expression.Constant(value.ToString())); // value is what we are checking to see if the column contains.
    LambdaExpression lambda = Expression.Lambda(condition, parameter);
    break;

但是,当列绑定的属性不是字符串类型(即 Int)时,这将失败,因为该类型Int没有“包含”方法。ToString()在调用“包含”之前,如何首先获取 memberAccess 的值?

注1:“memberAccess”所代表的属性类型在编译时是未知的。
注意 2:lambda 表达式最终被用于无法显式处理的 LINQ 2 实体查询中ToString()。(看看我在下面尝试了什么)

这是我尝试过的一种解决方案,但在 LINQ 表达式评估时失败,因为 LINQ 2 实体不支持ToString()

case WhereOperation.Contains:
    Expression stringValue = Expression.Call(memberAccess,
       typeof(object).GetMethod("ToString"));
    Expression condition = Expression.Call(
       stringValue, 
       typeof(string).GetMethod("Contains"),
       Expression.Constant(value.ToString()));
    LambdaExpression lambda = Expression.Lambda(condition, parameter);
    break;
4

1 回答 1

7

是的...由于安全原因,您不能在 Linq to Entities 中使用 ToString。不记得在哪里读的了。虽然它在 Linq2Sql 中得到支持。但是您可以使用 SqlFunctions.StringConvert。

我为你做了简单的扩展:

public static class ExpressionExtensions
{
    public static Expression<Func<T, bool>> AddContains<T>(this LambdaExpression selector, string value)
    {
        var mi = typeof (string).GetMethods().First(m => m.Name == "Contains" && m.GetParameters().Length == 1);


        var body = selector.GetBody().AsString();
        var x = Expression.Call(body, mi, Expression.Constant(value));

        LambdaExpression e = Expression.Lambda(x, selector.Parameters.ToArray());
        return (Expression<Func<T, bool>>)e;
    }

    public static Expression GetBody(this LambdaExpression expression)
    {
        Expression body;
        if (expression.Body is UnaryExpression)
            body = ((UnaryExpression)expression.Body).Operand;
        else
            body = expression.Body;

        return body;
    }

    public static Expression AsString(this Expression expression)
    {
        if (expression.Type == typeof (string))
            return expression;

        MethodInfo toString = typeof(SqlFunctions).GetMethods().First(m => m.Name == "StringConvert" && m.GetParameters().Length == 1 && m.GetParameters()[0].ParameterType == typeof(double?));
        var cast = Expression.Convert(expression, typeof(double?));
        return Expression.Call(toString, cast);
    }
}

用法:

        IQueryable<Foo> seed = ...;
        Expression<Func<Foo, int>> selector = x => x.Id;
        var expression = selector.AddContains<Foo>("3");
        seed.Where(expression);  
于 2013-05-09T11:49:51.923 回答