Reflexión.Emitir llamada de método genérico de clase base genérica
-
05-07-2019 - |
Pregunta
Subclasifico dinámicamente un tipo genérico (cumpliendo su contrato) que tiene un método genérico. Intento llamar a este método genérico pero el ensamblaje que produzco tiene errores. El reflector se bloquea al intentar abrir el ensamblaje y este fragmento de código no se ejecuta.
La excepción que obtengo es:
Se intentó cargar un programa con un formato incorrecto. (Excepción de HRESULT: 0x8007000B)
Cualquier ayuda sería apreciada. Voy a instalar PEVerify para echar un vistazo al código generado.
using System;
using System.Reflection;
using System.Reflection.Emit;
public class TestClass<TFirst>
{
public void TestMethod<TSecond>()
{
Console.WriteLine("It works");
}
}
public class Program
{
static void Main(string[] args)
{
// create a dynamic assembly and module
var assemblyName = new AssemblyName("DynamicAssembly");
var assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);
var module = assemblyBuilder.DefineDynamicModule("DynamicAssembly.dll");
// public class Voodoo.Dynamic.Class : TestClass<object>
var testClass = typeof(TestClass<>);
var testClassOfObject = testClass.MakeGenericType(typeof(object));
var typeBuilder = module.DefineType(
"Voodoo.Dynamic.Class",
TypeAttributes.Public | TypeAttributes.Class,
testClassOfObject);
// public void Run()
var methodBuilder = typeBuilder.DefineMethod("Run", MethodAttributes.Public);
// this.TestMethod<int>();
var testGeneric = testClass.GetMethod("TestMethod").MakeGenericMethod(typeof(int));
var il = methodBuilder.GetILGenerator();
il.Emit(OpCodes.Ldarg_0); // load this
il.Emit(OpCodes.Call, testGeneric);
il.Emit(OpCodes.Ret);
// bake it
var classType = typeBuilder.CreateType();
assemblyBuilder.Save("DynamicAssembly.dll");
// var instance = new Voodoo.Dynamic.Class();
var instance = Activator.CreateInstance(classType);
// instance.Run();
classType.GetMethod("Run").Invoke(instance, new object[] { });
}
}
Solución
En lugar de:
var testGeneric = testClass
.GetMethod("TestMethod")
.MakeGenericMethod(typeof(int));
Necesitas:
var testGeneric = testClassOfObject
.GetMethod("TestMethod")
.MakeGenericMethod(typeof(int));
porque testClass
es un tipo de clase genérico abierto y si intenta invocar un método, se bloqueará.
Licenciado bajo: CC-BY-SA con atribución
No afiliado a StackOverflow