![](/img/trans.png)
[英]Using Reflection.Emit to instantiate Generic Type with Generic Parameters
[英]Reflection.Emit with generic type = type is not generic
我正在使用Refletion.Emit,我有一个接口,一个抽象类和另一个类。 我要实现的是基于这两个类创建一个新类。
所以这是简单的界面:
public interface IHello()
{
string SayHello();
}
这是我的抽象类:
public abstract class Helloer<T> where T : IHello, new()
{
private readonly string text;
protected Helloer(string text)
{
this.text = text;
}
public string DoIt()
{
var t = new T();
return t.SayHello() + text;
}
}
第二类:
public class Howdy : IHello
{
public string SayHello() { return "Howdy"; }
}
现在,这是负责创建新类型HowdyHelloer的完整主要代码:
public static void Run()
{
var type = CreateHelloer(typeof(Howdy));
dynamic helloer = Activator.CreateInstance(type);
Console.WriteLine(helloer.DoIt());
}
public static Type CreateHelloer(Type hello)
{
var assemblyBuilder = GetAssemblyBuilder("MyAssembly");
var moduleBuilder = assemblyBuilder.DefineDynamicModule("MyModule");
var typeBuilder = moduleBuilder.DefineType(hello.Name + "Helloer", TypeAttributes.Public);
var parentType = typeof(Helloer<>).MakeGenericType(hello);
typeBuilder.SetParent(parentType);
Type[] types = new Type[1];
types[0] = typeof(string);
var parentCtorGeneric1 = typeof(Helloer<>).GetConstructor(BindingFlags.Instance | BindingFlags.NonPublic, null, types, null);
var parentCtor = TypeBuilder.GetConstructor(parentType, parentCtorGeneric1);
var ctor = typeBuilder.DefineConstructor(MethodAttributes.Public, CallingConventions.Standard, new Type[] { });
var ctorIl = ctor.GetILGenerator();
ctorIl.Emit(OpCodes.Ldstr, "Partner");
ctorIl.Emit(OpCodes.Call, parentCtor);
ctorIl.Emit(OpCodes.Ret);
return typeBuilder.CreateType();
}
public static AssemblyBuilder GetAssemblyBuilder(string name)
{
var assemblyName = new AssemblyName(name);
var domain = AppDomain.CurrentDomain;
AssemblyBuilder c = domain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.Run);
return c;
}
在线上:
var parentCtor = TypeBuilder.GetConstructor(parentType, parentCtorGeneric1);
我收到一个错误:““类型”必须包含TypeBuilder作为通用参数。” 有人可以帮我吗? 当我试图解决最近3天的问题时,什么也没做:/我做了研究,老实说,我没有发现将Emit与泛型抽象类结合使用的任何特定内容。
我在您的代码中看到至少两个错误
第一:
var parentCtor = TypeBuilder.GetConstructor(parentType, parentCtorGeneric1);
这里parentType
不是用TypeBuilder
创建的,所以如果要获取父构造器,只需从父类型获取它,例如
var parentCtorGeneric1 = parentType.GetConstructor(BindingFlags.Instance | BindingFlags.NonPublic, null, types, null);
第二:您错误地创建了容器代码,应该这样
var ctorIl = ctor.GetILGenerator();
ctorIl.Emit(OpCodes.Ldarg_0); // show where to load the following string
ctorIl.Emit(OpCodes.Ldstr, "Partner");
ctorIl.Emit(OpCodes.Call, parentCtorGeneric1);
ctorIl.Emit(OpCodes.Ret);
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.