我有一些代码使用IlGenerator.Emit来使用datareader创建和填充通用对象。它工作得很好但是我需要扩展它以在数据库字段名称包含下划线时填充简单的子对象。
例如,名为“Address_Line1”的数据库字段应填充属性Line1,该属性是Entity上Address属性的属性。在C#代码中基本上......
Entity.Address.Line1 = "value from reader";
我尝试编写c#代码并使用ILSpy来尝试识别我应该编写的IL代码,但是我一直遇到内存错误等。
下面的代码包含当前正在运行的IL代码,我已将代码尝试包含在注释中。任何人都可以帮助我吗?
public static DynamicBuilder<T> CreateBuilder(IDataRecord reader)
{
var result = new DynamicBuilder<T>();
var method = new DynamicMethod("DynamicCreate", typeof(T), new Type[] { typeof(IDataReader) }, typeof(T), true);
var generator = method.GetILGenerator();
generator.DeclareLocal(typeof(T));
generator.Emit(OpCodes.Newobj, typeof(T).GetConstructor(Type.EmptyTypes));
generator.Emit(OpCodes.Stloc_0);
var getValue = reader.GetType().GetMethod("get_Item", new Type[] { typeof(int) });
for (int i = 0; i < reader.FieldCount; i++)
{
var name = reader.GetName(i).Split('_'); // MY CODE
var propertyInfo = typeof(T).GetProperty(name[0]);
if (propertyInfo != null && propertyInfo.GetSetMethod() != null)
{
var endIfLabel = generator.DefineLabel();
generator.Emit(OpCodes.Ldarg_0);
generator.Emit(OpCodes.Ldc_I4, i);
generator.Emit(OpCodes.Callvirt, typeof(IDataRecord).GetMethod("IsDBNull"));
generator.Emit(OpCodes.Brtrue, endIfLabel);
generator.Emit(OpCodes.Ldloc_0);
generator.Emit(OpCodes.Ldarg_0);
generator.Emit(OpCodes.Ldc_I4, i);
generator.Emit(OpCodes.Callvirt, getValue);
if (propertyInfo.PropertyType.Name.ToLower().Contains("nullable"))
generator.Emit(OpCodes.Unbox_Any, GetNullableType(reader.GetFieldType(i)));
else
generator.Emit(OpCodes.Unbox_Any, reader.GetFieldType(i));
// START MY CODE TO GET THE SUB PROPERTY
if (name.Length > 1)
{
generator.Emit(OpCodes.Callvirt, propertyInfo.GetGetMethod());
propertyInfo = propertyInfo.PropertyType.GetProperty(name[1]);
}
// END MY CODE
generator.Emit(OpCodes.Callvirt, propertyInfo.GetSetMethod());
generator.MarkLabel(endIfLabel);
}
}
generator.Emit(OpCodes.Ldloc_0);
generator.Emit(OpCodes.Ret);
result.handler = (Load)method.CreateDelegate(typeof(Load));
return result;
}
答案 0 :(得分:3)
这样的代码:
static Entity DynamicCreate(IDataReader reader)
{
var entity = new Entity();
entity.Property = (int)reader[0];
return entity;
}
编译为IL,看起来与您发出的代码完全相同(省略了不重要的部分):
ldloc.0 // entity
ldarg.0 // reader
ldc.i4.0
callvirt System.Data.IDataRecord.get_Item
unbox.any System.Int32
callvirt UserQuery+Entity.set_Property
但是如果你添加第二个属性访问:
static Entity DynamicCreate(IDataReader reader)
{
var entity = new Entity();
entity.SubEntity.Property = (int)reader[0];
return entity;
}
然后IL看起来像这样:
ldloc.0 // entity
callvirt UserQuery+Entity.get_SubEntity
ldarg.0 // reader
ldc.i4.0
callvirt System.Data.IDataRecord.get_Item
unbox.any System.Int32
callvirt UserQuery+SubEntity.set_Property
请注意,对get_SubEntity
的调用是在ldloc.0
和ldarg.0
之间,而不是在代码中set_Property
之前,因此您必须在代码中将其移到太
你的代码不起作用的原因是IL是基于堆栈的语言:当你调用无参数实例方法(如属性getter)时,堆栈顶部的对象(这里是对象的结果) unbox.any
)将用作this
,这不是您想要的。基本上,您的代码尝试执行以下操作:
entity.Property = ((int)reader[0]).SubEntity;
答案 1 :(得分:-1)
我建议不要使用IL代码/ Emit,因为很难用它构建表达式。而是尝试使用新的Roslyn来生成委托。
以下是一些示例: https://github.com/dotnet/roslyn/wiki/Scripting-API-Samples