如何在C#中捕获表达式树?

内容来源于 Stack Overflow,并遵循CC BY-SA 3.0许可协议进行翻译与使用

  • 回答 (2)
  • 关注 (0)
  • 查看 (56)

我正在编写一些调试代码,在这些代码中,我希望同时显示原始表达式,并对任意表达式进行计算。

)例子:

IList<string> myString = /* Some string collection expression */

ShowMe(myString.Select(s => s.ToLower()));

我实现了ShowMe,类似于:

public void ShowMe(/* Not sure what has to go here */)
{
    /* Not sure what has to go here */
    Console.WriteLine(expression.ToString();
    IEnumerable result = expression.Evaluate(); // or something
    foreach(item in result)
    {
        Console.WriteLine(/* etc. */)
    }
}

结果将写入控制台,如下所示:

myString.Select(s => s.ToLower())

换句话说,我的ShowMe方法对表达式树进行操作,而不是对表达式的值进行操作,以便既可以显示给定的表达式,也可以显示计算的结果。

我不能简单地声明ShowMe

public void ShowMe(Expression expr)

但如果我声明:

public void ShowMe(Expression<Func<Enumerable>> expr)

sort-of工作-我必须使用lambda表达式调用我的方法,因此:

ShowMe(() => myString.Select(s => s.ToLower()))

我不想这么做

我相当肯定这是FluentAssertions可以做到的。例如:如果我执行以下测试代码:

(1 + 1).Should.Be(3)

我得到以下结果:

Expected (1 + 1) to be 3, but found 2.

FluentAssertion既计算了表达式(1+1),又捕获了表达式树,以便能够显示它计算的原始表达式。

我看不出是怎么做到的,但我也想做类似的事情。我该怎么做?

提问于
用户回答回答于

就其本身而言,任何方法都不可能做到这一点。

所有这些库只解析堆栈跟踪并提取文件名和行号。然后从给定行的源代码文件中提取表达式(这包括一些解析、验证)。

同样值得注意的是,如果源代码不存在、不可用,则不能显示表达式。

用户回答回答于

想出了一个可以接受的方案:

public static class ObjectHelper
{
    public static void ToConsole<T>(this IEnumerable<T> enumerable, Expression<Func<T,object>> expr)
        where T:class
    {
        var fn = expr.Compile();

        var result = enumerable.Select(s => fn(s));

        Console.WriteLine($"My data selected as {PrettyPrintExpression(expr)}");
        foreach(var element in result)
        {
            Console.WriteLine(/*  etc.  */);
        }
    }

    private static string PrettyPrintExpression(Expression<Func<T,object>> expr)
    {
        // Walk the expression tree to print as desired
    }
}

我可以调用:

IList<MyObject> list = /* etc. */
list.ToConsole(s => new{/* any members I want out of MyObject */});

扫码关注云+社区

领取腾讯云代金券