用于检查属性是否等于常量的表达式

时间:2014-12-17 16:34:17

标签: c# lambda expression

我试图合并两个工作表达式。

  • left:一个返回属性值

    的表达式
    Expression<Func<TimeSlot, Guid>> left = x => x.TimeSlotId;
    
  • right:一个返回常量值的表达式(变量guid的值)

    Expression<Func<TimeSlot, Guid>> right = Expression.Lambda<Func<TimeSlot, Guid>>
                        (Expression.Constant(guid, typeof(Guid)), input);
    

现在我想将这两个表达式打包成另一个表达式,它等同于两个结果。这有效:

    // given a TimeSlot slot which guid is equal to the constent
    bool eq1 = left.Compile()(slot) == right.Compile()(slot); // true

但这不起作用             ParameterExpression input = Expression.Parameter(typeof(TimeSlot));

        Expression<Func<TimeSlot, bool>> expression = Expression.Lambda<Func<TimeSlot, bool>>
            (Expression.Equal(left, right), input);

        bool eq2 = expression.Compile()(slot); // false

我完全不知道为什么这个结果会有所不同。特别是因为这个表达式的DebugView看起来像这样:

{Param_0 => (x => x.TimeSlotId == Param_0 => 351155b2-20a5-4c48-8722-ddf0e1f9055a)}

=

.Lambda #Lambda1<System.Func`2[SilverFit.Database.Model.TimeSlot,System.Boolean]>(SilverFit.Database.Model.TimeSlot $var1)
{
    .Lambda #Lambda2<System.Func`2[SilverFit.Database.Model.TimeSlot,System.Guid]> == .Lambda #Lambda3<System.Func`2[SilverFit.Database.Model.TimeSlot,System.Guid]>
}

.Lambda #Lambda2<System.Func`2[SilverFit.Database.Model.TimeSlot,System.Guid]>(SilverFit.Database.Model.TimeSlot $x) {
    $x.TimeSlotId
}

.Lambda #Lambda3<System.Func`2[SilverFit.Database.Model.TimeSlot,System.Guid]>(SilverFit.Database.Model.TimeSlot $var1) {
    .Constant<System.Guid>(351155b2-20a5-4c48-8722-ddf0e1f9055a)
}

有没有人知道创建表达式的正确方法,该表达式比较表达式和常量的结果?

1 个答案:

答案 0 :(得分:2)

您看到的调试输出正在告诉您究竟发生了什么,以及它为什么不起作用。特别是,您与等式表达式进行比较的事情不是表达式返回的值,而是表达式本身。

一个重要的线索是两个表达式都需要输入,但即使在调试输出中,您也可以看到该输入仅用于相等比较的一侧,并且仅作为第二个表达式的输入参数。

分别编译两个表达式实际上是恕我直言并不是一个糟糕的解决方案。您甚至可以缓存编译结果:

Func<TimeSlot, Guid> d1 = left.Compile(), d2 = right.Compile();
Func<TimeSlot, bool> d2 = x => d1(x) == d2(x);

但是如果你真的想把这个操作组成一个单独的表达式,你就可以这样做:

ParameterExpression param1 = Expression.Parameter(typeof(TimeSlot));
Expression<Func<TimeSlot, bool>> expression = Expression.Lambda<Func<TimeSlot, bool>>(
    Expression.Equal(
        Expression.Invoke(left, param1),
        Expression.Invoke(right, param1)),
    param1);

以下是演示该技术的完整代码示例:

class Program
{
    class A
    {
        public Guid Guid { get; private set; }

        public A()
        {
            Guid = Guid.NewGuid();
        }
    }

    static void Main(string[] args)
    {
        // Set up data values
        A a = new A();
        Guid guid = a.Guid;

        // Create expressions to be composed
        Expression<Func<A, Guid>> e1 = arg => arg.Guid, e2 = arg => guid;

        // Create lambda expression: invoke both expressions, compare the result
        ParameterExpression param1 = Expression.Parameter(typeof(A));
        Expression<Func<A, bool>> e3 = Expression.Lambda<Func<A, bool>>(
            Expression.Equal(
                Expression.Invoke(e1, param1),
                Expression.Invoke(e2, param1)),
            param1);

        // Compile to an actual delegate instance
        Func<A, bool> d1 = e3.Compile();

        // Check the result
        Console.WriteLine(d1(a));
    }
}