2013-03-27 45 views
4

我使用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异常。

所以,基本上,它不会让我得到只引用我的自定义类型的类型的构造函数,也不能在发射它的方法体之前最终确定类型。

难道真的不可能走出这个恶性循环吗?

+0

什么是例外细节? – 2013-03-27 21:08:20

+0

我向原始问题添加了有关异常的信息。 – Impworks 2013-03-27 21:12:54

+2

也许您没有阅读我在上一个问题中留下的最后一条评论。我会在这里重复它:** Reflection.Emit太弱,无法用于构建真正的编译器**对于诸如在LINQ查询中发出动态调用网站和表达式树的小玩具编译任务来说,这非常棒,但对于各种类型您将在编译器中遇到的问题将迅速超出其功能。使用CCI,而不是Reflection.Emit。 – 2013-03-27 21:18:37

回答

5

出于某种原因,您需要使用the static overload of GetConstructor()来执行此操作。在你的情况下,代码可能看起来像这样:

var ctor = TypeBuilder.GetConstructor(
    tupleType, typeof(Tuple<,>).GetConstructors().Single()); 
+0

哇!这确实有效!我甚至设法通过'GetMethod()'获得属性访问器。 – Impworks 2013-03-27 21:58:50