0

我正在尝试使用 Reflection.Emit 创建通用方法的这个简单示例,但是在调用 Invoke 时会导致异常,我找不到问题。

public class Program
{
    public static void Main(string[] args)
    {
        AppDomain appDomain = AppDomain.CurrentDomain;
        AssemblyName name = new AssemblyName("MyAssembly") { Version = new Version("1.0.0.0") };
        AssemblyBuilder ab = appDomain.DefineDynamicAssembly(name, AssemblyBuilderAccess.RunAndSave);
        ModuleBuilder mb = ab.DefineDynamicModule("MyModule", "MyAssembly.dll");
        TypeBuilder tb = mb.DefineType("Widget", TypeAttributes.Public);

        // Define a method builder
        MethodBuilder methodBuilder = tb.DefineMethod("MyGenericMethod", MethodAttributes.Public | MethodAttributes.Static);

        // Get generic type parameter builders 
        GenericTypeParameterBuilder[] genericParams =
            methodBuilder.DefineGenericParameters("TKey", "TValue");

        methodBuilder.SetSignature(typeof(int), null, null,
            genericParams,
            null, null);
        methodBuilder.DefineParameter(1, ParameterAttributes.None, "key");
        methodBuilder.DefineParameter(2, ParameterAttributes.None, "val");

        ILGenerator gen = methodBuilder.GetILGenerator();
        MethodInfo writeLineStr = typeof(Console).GetMethod("WriteLine", new[] { typeof(object) });
        gen.Emit(OpCodes.Ldarg_0);
        gen.Emit(OpCodes.Box, genericParams[0]);
        gen.Emit(OpCodes.Call, writeLineStr);
        gen.Emit(OpCodes.Ldarg_1);
        gen.Emit(OpCodes.Box, genericParams[1]);
        gen.Emit(OpCodes.Call, writeLineStr);
        gen.Emit(OpCodes.Ret);

        Type t = tb.CreateType();
        MethodInfo genMeth = t.GetMethod("MyGenericMethod").MakeGenericMethod(typeof(string), typeof(int));
        genMeth.Invoke(null, new object[] { "NumberKey", 100 });
    }
}

调用genMeth.Invoke时,抛出如下异常

Unhandled Exception: System.Reflection.TargetInvocationException: Exception has been thrown by the t
arget of an invocation. ---> System.InvalidProgramException: Common Language Runtime detected an inv
alid program.
   at Widget.MyGenericMethod[TKey,TValue](TKey key, TValue val)
   --- End of inner exception stack trace ---
   at System.RuntimeMethodHandle.InvokeMethod(Object target, Object[] arguments, Signature sig, Bool
ean constructor)
   at System.Reflection.RuntimeMethodInfo.UnsafeInvokeInternal(Object obj, Object[] parameters, Obje
ct[] arguments)
   at System.Reflection.RuntimeMethodInfo.Invoke(Object obj, BindingFlags invokeAttr, Binder binder,
 Object[] parameters, CultureInfo culture)
   at System.Reflection.MethodBase.Invoke(Object obj, Object[] parameters)
   at ReflectionEmitDemo14.Program.Main(String[] args) in d:\Projects\ReflectionEmitDemo\ReflectionE
mitDemo14\Program.cs:line 50
4

1 回答 1

3

您的方法是静态的,并且有两个参数:参数 0 和参数 1。但是在生成的 IL 代码中,您正在访问一个不存在的参数 2。

另一个问题是您试图将 Int32 值传递给需要 String 引用的方法。将两个参数装箱并将它们传递给接受 Object 引用的 WriteLine 重载。

您还声明了返回 Int32 的方法,但在返回之前您没有将任何内容放入堆栈。

尝试用 C# 编写方法,然后查看 C# 编译器发出的 IL(例如,使用 ILspy)。它应该如下所示:

ldarg.0
box !TKey
call void [mscorlib]System.Console::WriteLine(object)

ldarg.1
box !TValue
call void [mscorlib]System.Console::WriteLine(object)

ldc.i4.s 42
ret
于 2013-07-23T01:14:33.053 回答