2013-01-07 15 views
9

私は、生成された型にある汎用メソッドの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

+1

興味深いことに、 'methodInfo.GetGenericArguments()'はジェネリックパラメータである型を返しますが、 'methodInfo.IsGenericMethodDefinition'は確かに' false'です。 – svick

+0

ええそれを調べることは間違いなく一般的なものに見えるようにします。私の現在の解決策は、私のメソッドが一意の名前を持ち、 'Type'と' string'を渡し、 'type.GetMethod(...)'を呼び出すことですが、私は反映しないようにしたいと思います。 –

+0

それは今では解決されているようだ.net 4.7 –

答えて

3

は、私にはCLRの問題のように見えます。

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:それは、一般的なクラスGと非ジェネリッククラスN、汎用的な方法Mとそれぞれ、その後、非ジェネリッククラスからジェネリックメソッド定義を取得しようとする働き与えられた、ありますジェネリッククラスは、ジェネリックメソッドの定義ではありません(それがほとんどであり、Dが希望メソッド定義でどこにD.MakeGenericMethod(D.GetGenericArguments())です):

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 
+0

実際には、ちょうど具体的なタイプ(List .ConvertAll)で同じことをやってくれてありがとう、私はこれをMicrosoftにバグとして報告します。私は多くのことが起こるとは思わない:( –

+0

面白いのは、このILをコンパイルできないということです。 "エラー:トークン '[' – IllidanS4

0

問題は、理由に起因してldtoken method命令内にありますILが汎用メソッド定義を表現できない場合、CLRは間違ったメソッドをロードします。命令はildasmによって次のように逆コンパイルされます。

ldtoken method class [mscorlib]System.Reflection.MethodInfo class NameSpace.Generic`1<!T>::GM<[1]>() 

どちらも有効ではありません。その後、CLRは命令を壊し、その代わりにジェネリックメソッドインスタンスを独自の汎用パラメータからロードします。 (また、唯一の方法ハンドルではなく、宣言タイプを必要としなければならない)

DynamicMethod dyn = new DynamicMethod("", typeof(RuntimeMethodHandle), null); 
var il = dyn.GetILGenerator(); 
il.Emit(OpCodes.Ldtoken, typeof(GenClass<string>).GetMethod("GenMethod")); 
il.Emit(OpCodes.Ret); 
var handle = (RuntimeMethodHandle)dyn.Invoke(null, null); 
var m = MethodBase.GetMethodFromHandle(handle, typeof(GenClass<int>).TypeHandle); 

GetMethodFromHandleだけの宣言を設定します。

var methodInfoClosedOverBool = (methodInfo.IsGenericMethodDefinition ? methodInfo : methodInfo.GetGenericMethodDefinition()).MakeGenericMethod(typeof(bool)); 

は、より多くのテストのために、私は同じ問題を示す短いコードを作りました(通知<int>または<string>は問題ではありません)、間違ったことはありません。

関連する問題