我使用Reflection.Emit
,我想创建一个类型,这将是在C#中定义以下类型的等价:Reflection.Emit的和泛型类型
class A
{
public Tuple<A, int> GetValue(int x)
{
return new Tuple<A, int>(this, x);
}
}
诀窍是,我需要使用一个通用的使用我的自定义类型作为通用参数的BCL类型。
我用下面的代码片段搞乱:
var asmName = new AssemblyName("Test");
var access = AssemblyBuilderAccess.Run;
var asm = AppDomain.CurrentDomain.DefineDynamicAssembly(asmName, access);
var module = asm.DefineDynamicModule("Test");
var aType = module.DefineType("A");
var tupleType = typeof(Tuple<,>).MakeGenericType(aType, typeof(int));
var attrs = MethodAttributes.Public;
var method = aType.DefineMethod("GetValue", attrs, tupleType, new [] { typeof(int) });
var gen = method.GetILGenerator();
gen.Emit(OpCodes.Ldarg_0);
gen.Emit(OpCodes.Ldarg_1);
// here is the fail:
var ctor = tupleType.GetConstructor(new [] { typeof(int), aType });
gen.Emit(OpCodes.Newobj, ctor);
的调用GetConstructor
失败,出现以下异常:不支持指定的方法:
NotSupportedException异常。
所以,基本上,它不会让我得到只引用我的自定义类型的类型的构造函数,也不能在发射它的方法体之前最终确定类型。
难道真的不可能走出这个恶性循环吗?
什么是例外细节? – 2013-03-27 21:08:20
我向原始问题添加了有关异常的信息。 – Impworks 2013-03-27 21:12:54
也许您没有阅读我在上一个问题中留下的最后一条评论。我会在这里重复它:** Reflection.Emit太弱,无法用于构建真正的编译器**对于诸如在LINQ查询中发出动态调用网站和表达式树的小玩具编译任务来说,这非常棒,但对于各种类型您将在编译器中遇到的问题将迅速超出其功能。使用CCI,而不是Reflection.Emit。 – 2013-03-27 21:18:37