8

我找不到从 Expression<Func<T1,bool>> 转换为 Expression<Func<T2,bool>> 的方法。由于我使用了大量的反射,事实上,我真正需要的是一个接受类型参数并执行转换的方法。

public object Convert(Expression<Func<T1,bool>> expr, Type t);

T2 源自 T1

public class T1 {
     int FamilyId {get; set;}
}

public class T2 : T1 {
     ... other properties
}

我在基类上定义一个过滤器表达式

Expression<Func<T1,bool>> filter = p => p.FamilyId == [some value]

我想申请一个 List<T2>

4

3 回答 3

12

这是你要找的吗?该方法有两种风格:第一种让您将新的输入类型作为参数传递;第二个允许您将输入类型作为泛型参数传递并获得强类型的 LambdaExpression。

    public static LambdaExpression ChangeInputType<T, TResult>(Expression<Func<T, TResult>> expression, Type newInputType)
    {
        if (!typeof(T).IsAssignableFrom(newInputType))
            throw new Exception(string.Format("{0} is not assignable from {1}.", typeof(T), newInputType));
        var beforeParameter = expression.Parameters.Single();
        var afterParameter = Expression.Parameter(newInputType, beforeParameter.Name);
        var visitor = new SubstitutionExpressionVisitor(beforeParameter, afterParameter);
        return Expression.Lambda(visitor.Visit(expression.Body), afterParameter);
    }

    public static Expression<Func<T2, TResult>> ChangeInputType<T1, T2, TResult>(Expression<Func<T1, TResult>> expression)
    {
        if (!typeof(T1).IsAssignableFrom(typeof(T2)))
            throw new Exception(string.Format("{0} is not assignable from {1}.", typeof(T1), typeof(T2)));
        var beforeParameter = expression.Parameters.Single();
        var afterParameter = Expression.Parameter(typeof(T2), beforeParameter.Name);
        var visitor = new SubstitutionExpressionVisitor(beforeParameter, afterParameter);
        return Expression.Lambda<Func<T2, TResult>>(visitor.Visit(expression.Body), afterParameter);
    }

    public class SubstitutionExpressionVisitor : ExpressionVisitor
    {
        private Expression before, after;
        public SubstitutionExpressionVisitor(Expression before, Expression after)
        {
            this.before = before;
            this.after = after;
        }
        public override Expression Visit(Expression node)
        {
            return node == before ? after : base.Visit(node);
        }
    }
于 2013-03-04T23:51:01.603 回答
0

你的要求是非常不明智的。编译器如何知道 T1 是否可以转换为 T2?似乎要求可怕的运行时错误,即使它是可能的。*

(*我认为这是不可能的,因为您试图将反射与嵌套的泛型类型结合起来。)

于 2013-03-04T23:06:57.543 回答
0

看起来您想要组合 2 个表达式 -T2转换T1,而不是调用给expr定结果。

这个问题一般讨论组合两个表达式(Expression<Func<T, bool>>)。对于您的情况,我认为您需要Expression.Call来构造转换表达式,而不是再次调用具有转换结果的原始表达式。

于 2013-03-04T23:09:22.867 回答