表达式选择方法

时间:2015-10-20 01:47:06

标签: c# lambda moq

我正在尝试根据Dynamically calling Moq Setup() at runtime

使用反射动态配置Moq

在他们使用的例子中

var body = Expression.PropertyOrField( parameter, "ExampleProperty" );

创建表达式以在对象上选择所需的属性,然后使用该选择器构造lambda表达式。

我希望我的选择器选择一种方法,例如我想动态构造以下内容:

mock.Setup(m => m.MyMethod()).Returns(1); 

我尝试过使用:

var body = Expression.PropertyOrField(parameter, "MyMethod");

var body = Expression.MakeMemberAccess(parameter, typeof(T).GetMethod("MyMethod"));

但两者似乎只适用于属性或字段,我可以使用不同的选择器来选择方法吗?

以下完整代码:

var mock = new Mock<T>();
var parameter = Expression.Parameter(typeof(T));

if (typeof(T).GetMethod("MyMethod") != null)
{
    var body = Expression.PropertyOrField(parameter, "MyMethod");
    var lambdaExpression = Expression.Lambda<Func<T, int>>(body, parameter);
    mock.Setup(lambdaExpression).Returns(0);
}

1 个答案:

答案 0 :(得分:2)

虽然我无法在此处准确访问Moq,但我可以向您展示如何构建调用方法的lambda表达式

private void DoIt<T>()
{
    var mock = new Mock<T>();
    var parameter = Expression.Parameter(typeof(T));
    var methodInfo = typeof(T).GetMethod("MyMethod"); //Find the method "MyMethod" on type "T"
    if (methodInfo != null)
    {
        var body = Expression.Call(parameter, methodInfo);
        var lambdaExpression = Expression.Lambda<Func<T, int>>(body, parameter);
        //At this point, lambdaExpression is:
        //Param_0 => Param_0.MyMethod()
        mock.Setup(lambdaExpression).Returns(0);
    }
}

class MyClass 
{
    public int MyMethod() 
    {
        return 5;
    }
}

请注意,typeof(T).GetMethod("MyMethod")不是非常具体。您可以使用大量重载(或利用GetMethods和过滤器),这些重载将允许您按名称,返回类型和参数类型指定方法。

另外,请注意Expression.Lambda<Func<T, int>>仅适用于没有返回int的参数的方法。根据负责确定方法的人员,您可能也希望具有此可配置性。