9

MethodInfo生成された型で見つかったジェネリックメソッドのを使用するコードがいくつかあります。いくつかの反射を避けるために、私はコードに使用させます

ldtoken Method
ldtoken Type
call GetMethodFromHandle(RuntimeMethodHandle,RunTimeTypeHandle)

コンパイル時にMethodInfoを生成するパターン。

ただし、me​​thodInfoがジェネリック型に属し、それ自体がジェネリックメソッドである場合は、問題が発生します。これは、methodInfoのオープンバージョンを発行するGMを単純に生成するコードです。特定のタイプでメソッドを閉じようとするよりもメソッドを取得するために呼び出すと、複雑な例外が発生します::

System.Reflection.MethodInfoGM[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-ジェネリック型

4

2 に答える 2

3

ILを手で書いてilasmを使用すると同じことが起こるので、私にはCLRの問題のように見えます。つまり、ジェネリック クラス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
于 2013-01-07T19:28:40.410 に答える