我有一些代码使用MethodInfo
在生成的类型上找到的泛型方法。为了避免一些反思,我让代码使用
ldtoken Method
ldtoken Type
call GetMethodFromHandle(RuntimeMethodHandle,RunTimeTypeHandle)
在编译时生成 MethodInfos 的模式。
但是,如果 methodInfo 属于泛型类型并且本身是泛型方法,那么事情就会变得很糟糕。下面是一些简单地生成 GM 的代码,该 GM 发出其 methodInfo 的开放版本。如果我调用它来检索方法而不是尝试通过特定类型关闭它,我会得到一个令人困惑的异常::
System.Reflection.MethodInfo
GM[M]()
不是 GenericMethodDefinition。MakeGenericMethod 只能在 MethodBase.IsGenericMethodDefinition 为 true 的方法上调用。
这是相关代码::
var aBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(new AssemblyName("Test"), AssemblyBuilderAccess.RunAndSave);
var mBuilder = aBuilder.DefineDynamicModule(aBuilder.GetName().Name, true);
var typeBuilder = mBuilder.DefineType("NameSpace.Generic`1",TypeAttributes.AutoClass | TypeAttributes.Sealed | TypeAttributes.Public,typeof(object));
var TypeGenerics = typeBuilder.DefineGenericParameters(new[] { "T" });
var methodBuilder = typeBuilder.DefineMethod("GM", MethodAttributes.Public | MethodAttributes.Static | MethodAttributes.HideBySig);
var methodGenerics = methodBuilder.DefineGenericParameters(new[] { "M" });
methodBuilder.SetSignature(typeof(MethodInfo), null, null, Type.EmptyTypes, null, null);
var ilgenerator = methodBuilder.GetILGenerator();
var typeBuilderClosedOverT = typeBuilder.MakeGenericType(TypeGenerics);
ilgenerator.Emit(OpCodes.Ldtoken, methodBuilder);
ilgenerator.Emit(OpCodes.Ldtoken, typeBuilderClosedOverT);
ilgenerator.Emit(OpCodes.Call,
typeof(MethodBase).GetMethod(
"GetMethodFromHandle",
BindingFlags.Public | BindingFlags.Static,
null,
new[] { typeof(RuntimeMethodHandle), typeof(RuntimeTypeHandle) },
null
)
);
ilgenerator.Emit(OpCodes.Castclass,typeof(MethodInfo));
ilgenerator.Emit(OpCodes.Ret);
var bakedType = typeBuilder.CreateType();
var methodInfo = bakedType.MakeGenericType(typeof(int)).GetMethod("GM").MakeGenericMethod(typeof(bool)).Invoke(null, null) as MethodInfo;
var methodInfoClosedOverBool = methodInfo.MakeGenericMethod(typeof(bool));
似乎我的代码唯一一次出错是它是否是非泛型类型的泛型方法。如果代码被重写,以使其关于普通类型的普通方法,或普通类型的泛型方法,或泛型类型的普通方法,它都可以工作。只有两者的结合才会导致错误。难道我做错了什么?
我提交了一个关于这个问题的错误: https ://connect.microsoft.com/VisualStudio/feedback/details/775989/clr-cannot-emit-a-token-for-an-open-generic-method-on-a-泛型