2013-03-27 44 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