4

我目前正在对 Moq 框架进行扩展,以模拟非虚拟方法的实现。我目前已经通过获取原始方法的方法句柄并将其与用户定义的 Func 的指针交换来进行此工作。

我仍然遇到的一个问题是,当我在 Moq 内部代码(在使用泛型的类中)创建 Func 时,我遇到了 RuntimeHelpers.PrepareMethod 的问题。(在我们执行指针交换之前,需要准备好 Func)。

当我在普通类(例如程序)中创建完全相同的 Func 时,一切正常。

进一步调查该问题可以追溯到调用类是否具有通用参数。

抛出异常:

An unhandled exception of type 'System.ArgumentException' occurred in mscorlib.dll

Additional information: The given generic instantiation was invalid.

我在以下代码块中隔离了问题:

class Program
{
    static void Main(string[] args)
    {
        new WithoutGeneric().GoExecute();
        new WithGeneric<string>().GoExecute();
    }
}

public class WithoutGeneric
{
    public void GoExecute()
    {
        //Works fine
        StaticMethods.PrepareThisFunc(() => "Test");
    }
}

public class WithGeneric<T>
{
    public void GoExecute()
    {
        //Breaks
        StaticMethods.PrepareThisFunc(() => "Test");
    }
}

public static class StaticMethods
{
    public static void PrepareThisFunc(Func<string> theFunc)
    {
        RuntimeHelpers.PrepareMethod(theFunc.Method.MethodHandle);
    }
}

我还查看了当前的开源 CoreCLR 代码,但无法找出问题所在。

CoreCLR: https ://github.com/dotnet/coreclr/blob/master/src/vm/reflectioninvocation.cpp

异常被抛出: 2435, 2444, 2447

有谁知道如何解决这个异常?

4

1 回答 1

6

CLR 对类中的类型参数是未知的WithGeneric,它需要知道它才能创建调用图。

这样做可以解决问题:

class Program
{
    static void Main(string[] args)
    {
        new WithoutGeneric().GoExecute();
        new WithGeneric<string>().GoExecute();
    }
}

public class WithoutGeneric
{
    public void GoExecute()
    {
        //Works fine
        StaticMethods.PrepareThisFunc1(() => "Test");
    }
}

public class WithGeneric<T>
{
    public void GoExecute()
    {
        //Works fine
        StaticMethods.PrepareThisFunc2(() => "Test");
    }
}

public static class StaticMethods
{
    public static void PrepareThisFunc1(Func<string> theFunc)
    {
        RuntimeHelpers.PrepareMethod(theFunc.Method.MethodHandle);
    }

    public static void PrepareThisFunc2(Func<string> theFunc)
    {
        RuntimeHelpers.PrepareMethod(theFunc.Method.MethodHandle, new[] { typeof(string).TypeHandle });
    }
}
于 2015-07-31T20:45:04.617 回答