Ответ 1
Если вы используете Reflection.Emit, вам действительно нужно взять копию Reflection.Emit language add-in для Reflector. Несмотря на то, что он не идеален, он должен получить по крайней мере 95% пути к любому испускаемому коду.
Мне нужно сгенерировать класс, используя 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);
}
}
Если вы используете Reflection.Emit, вам действительно нужно взять копию Reflection.Emit language add-in для Reflector. Несмотря на то, что он не идеален, он должен получить по крайней мере 95% пути к любому испускаемому коду.
У меня нет компилятора, но что-то вроде этого должно работать:
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();
Я верю AutoMapper и/или LinFu сделает это за вас. Вы можете создать экземпляр интерфейса с помощью AutoMapper, я это сделал.
Вы забыли вернуть 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));
}
Кажется, вы хотите быстро получить доступ к свойствам объекта по имени без отражения во время выполнения. Используя 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 и динамического типа?