2016-12-30 73 views
5

我要動態地生成類型是這樣的:反射發出

public class Sample 
{ 
    Sample Parent { get; set; } 
    public Sample(Sample parent) 
    { 
     Parent = parent; 
    } 

    public int Depth 
    { 
     get 
     { 
      if (Parent == null) 
       return -1; 
      else 
       return Parent.Depth + 1; 
     } 
    } 
} 

我寫的代碼是:

 const string assemblyName = "SampleAssembly"; 
     const string parentPproperty = "Parent"; 
     const string depthProperty = "Depth"; 
     const string typeName = "Sample";  
     const string assemblyFileName = assemblyName + ".dll"; 

     AppDomain domain = AppDomain.CurrentDomain; 
     AssemblyBuilder assemblyBuilder = domain.DefineDynamicAssembly(new AssemblyName(assemblyName), AssemblyBuilderAccess.RunAndSave); 
     ModuleBuilder moduleBuilder = assemblyBuilder.DefineDynamicModule(assemblyName, assemblyFileName); 
     TypeBuilder typeBuilder = moduleBuilder.DefineType(typeName, TypeAttributes.Public); 
     FieldBuilder parentField = typeBuilder.DefineField($"_{parentPproperty}", typeBuilder, FieldAttributes.Private); 
     PropertyBuilder propertyBuilder = typeBuilder.DefineProperty(parentPproperty, PropertyAttributes.None, parentField.FieldType, Type.EmptyTypes); 
     MethodAttributes getSetAttr = MethodAttributes.Public | 
     MethodAttributes.SpecialName | MethodAttributes.HideBySig; 

     MethodBuilder getParentMethod = typeBuilder.DefineMethod($"get_{propertyBuilder.Name}", getSetAttr, parentField.FieldType, Type.EmptyTypes); 
     ILGenerator il = getParentMethod.GetILGenerator(); 
     il.Emit(OpCodes.Ldarg_0); 
     il.Emit(OpCodes.Ldfld, parentField); 
     il.Emit(OpCodes.Ret); 
     propertyBuilder.SetGetMethod(getParentMethod); 

     MethodBuilder setParentMethod = typeBuilder.DefineMethod($"set_{propertyBuilder.Name}", qetSetAttr, null, Type.EmptyTypes); 
     il = setParentMethod.GetILGenerator(); 
     il.Emit(OpCodes.Ldarg_0); 
     il.Emit(OpCodes.Ldarg_1); 
     il.Emit(OpCodes.Stfld, parentField); 
     il.Emit(OpCodes.Ret); 
     propertyBuilder.SetSetMethod(setParentMethod); 


     parentField = typeBuilder.DefineField($"_{depthProperty}", typeBuilder, FieldAttributes.Private); 
     propertyBuilder = typeBuilder.DefineProperty(depthProperty, PropertyAttributes.None, parentField.FieldType, Type.EmptyTypes); 
     MethodBuilder getDepthMethod = typeBuilder.DefineMethod($"get_{depthProperty}", getSetAttr , parentField.FieldType, Type.EmptyTypes); 
     il = getDepthMethod.GetILGenerator(); 
     LocalBuilder lb = il.DeclareLocal(typeof(bool)); 

     il.Emit(OpCodes.Ldarg_0); 
     il.Emit(OpCodes.Call, getParentMethod); 
     il.Emit(OpCodes.Ldnull); 
     il.Emit(OpCodes.Ceq); 
     il.Emit(OpCodes.Stloc_0); 
     il.Emit(OpCodes.Ldloc_0); 
     il.Emit(OpCodes.Brfalse_S); 
     il.Emit(OpCodes.Ldc_I4_1); 
     il.Emit(OpCodes.Stloc_1); 
     il.Emit(OpCodes.Br_S); 
     il.Emit(OpCodes.Ldarg_0); 
     il.Emit(OpCodes.Call, getParentMethod); 
     il.Emit(OpCodes.Callvirt, getDepthMethod); 
     il.Emit(OpCodes.Ldc_I4_1); 
     il.Emit(OpCodes.Add); 
     il.Emit(OpCodes.Stloc_1); 
     il.Emit(OpCodes.Br_S); 
     il.Emit(OpCodes.Ldloc_1); 
     il.Emit(OpCodes.Ret); 
     propertyBuilder.SetGetMethod(getDepthMethod); 

     ConstructorBuilder constructor = typeBuilder.DefineConstructor(MethodAttributes.Public, CallingConventions.HasThis, new Type[] { typeBuilder }); 
     il= constructor.GetILGenerator();    
     il.Emit(OpCodes.Ldarg_0); 
     il.Emit(OpCodes.Call, typeof(object).GetConstructor(Type.EmptyTypes)); 
     il.Emit(OpCodes.Ldarg_0); 
     il.Emit(OpCodes.Ldarg_1); 
     il.Emit(OpCodes.Call, setParentMethod);   
     il.Emit(OpCodes.Ret); 

     Type type = typeBuilder.CreateType(); 
     var obj1 = Activator.CreateInstance(type, null); 

     var obj2 = Activator.CreateInstance(type, obj1); 

     assemblyBuilder.Save(assemblyFileName); 

我想我在建築構造和深度屬性的getter方法有問題。

請幫我解決這個問題。

實例未創建。

謝謝

回答

3

你的代碼有幾個問題。

在你父母的setter方法,你錯過了聲明參數:

MethodBuilder setParentMethod = typeBuilder.DefineMethod($"set_{propertyBuilder.Name}", getSetAttr, null, new [] { propertyBuilder.PropertyType }); 

您在這裏宣佈冗餘支持字段,只是刪除這一行:

parentField = typeBuilder.DefineField($"_{depthProperty}", typeBuilder, FieldAttributes.Private); 

你走向深沉屬性的錯誤類型,它必須是int類型的:

propertyBuilder = typeBuilder.DefineProperty(depthProperty, PropertyAttributes.None, typeof(int), Type.EmptyTypes); 
MethodBuilder getDepthMethod = typeBuilder.DefineMethod($"get_{depthProperty}", getSetAttr, propertyBuilder.PropertyType, Type.EmptyTypes); 

您是ge的IL代碼爲您的計算屬性打理好似調試代碼,我將其替換爲發佈代碼。你也發射不完整的轉移指令,你應該通過目標標籤作爲第二個參數,看看這個工作methodbody:

il = getDepthMethod.GetILGenerator(); 

var notNullLabel = il.DefineLabel(); 

il.Emit(OpCodes.Ldarg_0); 
il.Emit(OpCodes.Call, getParentMethod); 
il.Emit(OpCodes.Brtrue_S, notNullLabel); 
il.Emit(OpCodes.Ldc_I4_M1); 
il.Emit(OpCodes.Ret); 
il.MarkLabel(notNullLabel); 
il.Emit(OpCodes.Ldarg_0); 
il.Emit(OpCodes.Call, getParentMethod); 
il.Emit(OpCodes.Callvirt, getDepthMethod); 
il.Emit(OpCodes.Ldc_I4_1); 
il.Emit(OpCodes.Add); 
il.Emit(OpCodes.Ret); 

最後但並非最不重要的,你必須要包裝你的空參數對象數組等等該激活器可以找到預期的構造過載:

var obj1 = Activator.CreateInstance(type, new object[] { null }); 
+0

謝謝thehennyy,您的意見和糾正。這對你很好。 –