Использование Reflection.Emit для создания класса, реализующего интерфейс

Мне нужно сгенерировать класс, используя Reflection.Emit, который реализует следующий интерфейс.

public interface IObject
{
    T Get<T>(string propertyName); 
}

Есть ли у кого-нибудь пример того, как я буду использовать следующее в качестве простого тестового примера?

class GeneratedObject : IObject
{
    public T Get<T>(string propertyName)
    {
        // this is the simplest possible implementation
        return default(T);
    }
}

Ответы

Ответ 1

Если вы используете Reflection.Emit, вам действительно нужно взять копию Reflection.Emit language add-in для Reflector. Несмотря на то, что он не идеален, он должен получить по крайней мере 95% пути к любому испускаемому коду.

Ответ 2

У меня нет компилятора, но что-то вроде этого должно работать:

var aName = new AssemblyName("temp");
var appDomain = Threading.Thread.GetDomain();
var aBuilder = appDomain.DefineDynamicAssembly(aName, AssemblyBuilderAccess.Run);
var mBuilder = aBuilder.DefineDynamicModule(aName.Name);
var tBuilder = mBuilder.DefineType("GeneratedObject", TypeAttributes.Public | TypeAttributes.Class);
tBuilder.AddInterfaceImplementation(typeof(IObject));
var methBuilder = tBuilder.DefineMethod("Get", MethodAttributes.Public | MethodAttributes.Final | MethodAttributes.Virtual);
var typeParam = mb.DefineGenericParameters(new string[] { "T" })[0];
methBuilder.SetParameters(new Type[] { typeof(string) });
methBuilder.SetReturnType(typeParam);
var ilg = methBuilder.GetILGenerator();
let lBuilder = ilg.DeclareLocal(typeParam);
ilg.Emit(OpCodes.Ldloca_S, (byte)0);
ilg.Emit(OpCodes.Initobj, typeParam);
ilg.Emit(OpCodes.Ldloc_0);
ilg.Emit(OpCodes.Ret);
var generatedType = tBuilder.CreateType();

Ответ 3

Я верю AutoMapper и/или LinFu сделает это за вас. Вы можете создать экземпляр интерфейса с помощью AutoMapper, я это сделал.

Ответ 4

Вы забыли вернуть BOX:

internal delegate object FastConstructorHandler(object[] paramters);

    private static FastConstructorHandler CreateDelegate(Type Tipo)
    {
        DynamicMethod dynamicMethod = new DynamicMethod(string.Empty,
            typeof(object), new Type[] { typeof(object[]) }, Tipo.Module, false);

        ILGenerator ilg = dynamicMethod.GetILGenerator();

        ilg.DeclareLocal(Tipo);
        ilg.Emit(OpCodes.Ldloca_S, (byte)0);
        ilg.Emit(OpCodes.Initobj, Tipo);
        ilg.Emit(OpCodes.Ldloc_0);
        ilg.Emit(OpCodes.Box, Tipo);
        ilg.Emit(OpCodes.Ret);

        return (FastConstructorHandler)dynamicMethod.CreateDelegate(typeof(FastConstructorHandler));
    }

Ответ 5

Кажется, вы хотите быстро получить доступ к свойствам объекта по имени без отражения во время выполнения. Используя Yappi и свой класс Property < > , вы можете реализовать данный интерфейс следующим образом:

class GeneratedObject : IObject
{
    public string Value { get { return "Test"; } }

    public T Get<T>(string propertyName)
    {
        return Property<GeneratedObject>.Get<T>(this,propertyName);
    }
}

а затем используйте его следующим образом:

IObject obj = new GeneratedObject();
var value = obj.Get<String>("Value"); //value contains "Test"

Вам все еще нужна конструкция IObject и динамического типа?