4

我有一些代码使用 IlGenerator.Emit 使用数据读取器创建和填充通用对象。它很好用,但是当数据库字段名称包含下划线时,我需要扩展它以填充简单的子对象。

例如,名为“Address_Line1”的数据库字段应填充属性 Line1,它是实体上地址属性的属性。在 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;
}
4

2 回答 2

4

像这样的代码:

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.0and之间ldarg.0,而不是set_Property像在您的代码中那样,所以您也必须将它移到您的代码中。

您的代码不起作用的原因是 IL 是一种基于堆栈的语言:当您调用无参数实例方法(如属性 getter)时,堆栈顶部的对象(这里是 的结果unbox.any)将是用作它的this,这不是你想要的。基本上,您的代码尝试执行以下操作:

entity.Property = ((int)reader[0]).SubEntity;
于 2014-09-28T19:11:14.800 回答
-2

我建议根本不要使用 IL 代码/Emit,因为很难用它来构建表达式。而是尝试使用新的 Roslyn 来生成代表。

以下是一些示例: https ://github.com/dotnet/roslyn/wiki/Scripting-API-Samples

于 2015-12-27T20:36:11.207 回答