私は、生成された型にある汎用メソッドのMethodInfo
を使用するコードをいくつか持っています。反射を避けるために、コードを使用します。TypeBuilderがジェネリックメソッドインフォメーションをジェネリックメソッドとして生成しないのはなぜですか?
コンパイル時にMethodInfosを生成するパターン。
しかし、methodInfoがジェネリック型に属していて、それ自体がジェネリックメソッドである場合、状況は悪化します。 ここには、単にmethodInfoのオープンバージョンを発行するGMを生成するコードがあります。 私は困惑例外を取得特定の型の上に、それを閉じよう以外の方法を取得するには、それを呼び出した場合::
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));
それは非ジェネリック型のgenericmethodだ場合は私のコードのネジのみタイムアップがあるようです。通常の型の通常のメソッド、または通常の型の汎用メソッド、または汎用型の通常のメソッドがすべて機能するように、コードが書き換えられるとします。エラーの原因となるのは両方の組み合わせだけです。私は何か間違っているのですか?
私はこの問題についてのバグ提出:あなたが手でILを書いて、ILASMを使用した場合、同じことが起こるので https://connect.microsoft.com/VisualStudio/feedback/details/775989/clr-cannot-emit-a-token-for-an-open-generic-method-on-a-generic-type
興味深いことに、 'methodInfo.GetGenericArguments()'はジェネリックパラメータである型を返しますが、 'methodInfo.IsGenericMethodDefinition'は確かに' false'です。 – svick
ええそれを調べることは間違いなく一般的なものに見えるようにします。私の現在の解決策は、私のメソッドが一意の名前を持ち、 'Type'と' string'を渡し、 'type.GetMethod(...)'を呼び出すことですが、私は反映しないようにしたいと思います。 –
それは今では解決されているようだ.net 4.7 –