2016-09-15 9 views
1

Я пытаюсь создать этот класс C# с помощью System.Reflection.Emit.Создайте экземпляр класса с помощью Reflection.Emmit

private class MyTestData : IMyClonable 
{ 
    private int _testValue = 0; 
    public int testValue 
    { 
     get { return _testValue; } 
     set { _testValue = value; } 
    } 

    public IMyClonable Clone() 
    { 
     MyTestData clone = new MyTestData(); 
     clone._testValue = _testValue; 
     return clone ; 
    } 
} 

Этот класс должен быть создан из этого интерфейса:

public interface IMyTestData : IMyClonable 
{ 
    int testValue { get; set; } 
} 

Я уже сделал код, который генерирует свойства и штраф этого произведения. Но я застрял, когда пытаюсь создать метод Clone(). Я не знаю, как создать экземпляр этого класса и сохранить его в локальной переменной. Вот код для создания метода Clone():

private static void MakeCloneMethod(Type componentType, TypeBuilder typeBuilder) 
{ 
    ConstructorBuilder ctor = 
     typeBuilder.DefineDefaultConstructor(MethodAttributes.Public); 

    MethodInfo cloneMethod = typeof(IMyClonable).GetMethod("Clone"); 
    MethodAttributes methodIlAttributes = 
     (cloneMethod.Attributes & ~MethodAttributes.Abstract) | MethodAttributes.Final; 

    MethodBuilder cloneMthdBldr = typeBuilder.DefineMethod(
     "Clone", methodIlAttributes, typeof(IMyClonable), new Type[] { }); 
    ILGenerator ilgen = cloneMthdBldr.GetILGenerator(); 

    LocalBuilder returnValue = ilgen.DeclareLocal(typeBuilder.AsType()); 

    ilgen.Emit(OpCodes.Newobj, ctor); 
    ilgen.Emit(OpCodes.Stloc_S, returnValue); 

    CloneProperties(componentType, ilgen); 

    ilgen.Emit(OpCodes.Ldloc_S); 
    ilgen.Emit(OpCodes.Ret); 

    typeBuilder.DefineMethodOverride(cloneMthdBldr, cloneMethod); 
} 

private static void CloneProperties(Type componentType, ILGenerator ilgen) 
{ 
    PropertyInfo[] allProperties = GetPublicProperties(componentType); 
    foreach (PropertyInfo propInfo in allProperties) 
    { 
     ilgen.Emit(OpCodes.Ldarg_0); 
     ilgen.Emit(OpCodes.Ldfld, builders[propInfo]); 
     ilgen.Emit(OpCodes.Stfld, builders[propInfo]); 
     ilgen.Emit(OpCodes.Ldloc_0); 
    } 
} 

я получаю System.InvalidProgramException когда я пытаюсь вызвать метод Clone(). Даже если я прокомментирую вызов метода CloneProperties(). Что я делаю неправильно?

+0

Вы пытались сохранить свою сборку на диск и запустить на ней вечность? – svick

+0

to svick: Я нашел свои ошибки благодаря вам! –

ответ

0

Я понял! Вот рабочий код:

private static void MakeCloneMethod(Type componentType, TypeBuilder typeBuilder) 
{ 
    Type thisType = typeBuilder.AsType(); 
    Type itype = typeof(IMyClonable); 
    MethodInfo cloneMthd = itype.GetMethod("Clone"); 

    MethodBuilder cloneMthdBldr = typeBuilder.DefineMethod(
     cloneMthd.Name, cloneMthd.Attributes & ~MethodAttributes.Abstract, 
     itype, new Type[] {}); 
    typeBuilder.DefineMethodOverride(cloneMthdBldr, cloneMthd); 

    ILGenerator gen = cloneMthdBldr.GetILGenerator(); 

    LocalBuilder returnValue = gen.DeclareLocal(thisType); 

    gen.Emit(OpCodes.Newobj, typeBuilder.DefineDefaultConstructor(MethodAttributes.Public)); 
    gen.Emit(OpCodes.Stloc_S, returnValue); 

    PropertyInfo[] allProperties = GetPublicProperties(componentType); 
    foreach (PropertyInfo propInfo in allProperties) 
    { 
     gen.Emit(OpCodes.Ldloc_S, returnValue); 
     gen.Emit(OpCodes.Ldarg_0); 
     gen.Emit(OpCodes.Ldfld, builders[propInfo]); 
     gen.Emit(OpCodes.Stfld, builders[propInfo]); 
    } 

    gen.Emit(OpCodes.Ldloc_S, returnValue); 
    gen.Emit(OpCodes.Ret); 
} 

Благодаря Svick!