gpt4 book ai didi

c# - 为什么 TypeBuilder 生成的泛型方法信息不是泛型方法?

转载 作者:太空狗 更新时间:2023-10-29 17:57:24 30 4
gpt4 key购买 nike

我有一些代码使用在生成类型上找到的泛型方法的 MethodInfo。为了避免一些反射(reflection),我让代码使用

ldtoken Method
ldtoken Type
call GetMethodFromHandle(RuntimeMethodHandle,RunTimeTypeHandle)

在编译时生成 MethodInfos 的模式。

但是,如果 methodInfo 属于泛型类型并且它本身是一个泛型方法,事情就会变得很麻烦。下面是一些简单地生成 GM 的代码,该 GM 发出其 methodInfo 的开放版本。如果我调用它来检索方法而不是尝试通过特定类型关闭它,我会得到一个令人困惑的异常::

System.Reflection.MethodInfo GM[M]() is not a GenericMethodDefinition. MakeGenericMethod may only be called on a method for which MethodBase.IsGenericMethodDefinition is 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-generic-type

最佳答案

在我看来这像是 CLR 问题,因为如果您手动编写 IL 并使用 ilasm,也会发生同样的事情。也就是说,给定一个泛型类 G 和一个非泛型类 N,每个类都有一个泛型方法 M,然后尝试获取泛型方法定义来自非通用类作品:

ldtoken    method void class N::M<[1]>()
ldtoken class N<!T>
call class [mscorlib]System.Reflection.MethodBase [mscorlib]
System.Reflection.MethodBase::GetMethodFromHandle(
valuetype [mscorlib]System.RuntimeMethodHandle,
valuetype [mscorlib]System.RuntimeTypeHandle)
castclass [mscorlib]System.Reflection.MethodInfo
ret

但是从泛型类返回的 MethodInfo 不是泛型方法定义(但它几乎是;它是 D.MakeGenericMethod(D.GetGenericArguments()) 其中 D 就是你想要的方法定义):

ldtoken    method void class G`1<!T>::M<[1]>()
ldtoken class G`1<!T>
call class [mscorlib]System.Reflection.MethodBase [mscorlib]
System.Reflection.MethodBase::GetMethodFromHandle(
valuetype [mscorlib]System.RuntimeMethodHandle,
valuetype [mscorlib]System.RuntimeTypeHandle)
castclass [mscorlib]System.Reflection.MethodInfo
ret

关于c# - 为什么 TypeBuilder 生成的泛型方法信息不是泛型方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14200398/

30 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com