4

IlGenerator.Emit を使用して、datareader を使用して汎用オブジェクトを作成および設定するコードがあります。うまく機能しますが、データベース フィールド名にアンダースコアが含まれている場合は、単純な子オブジェクトを設定するように拡張する必要があります。

たとえば、「Address_Line1」という名前のデータベース フィールドは、エンティティの Address プロパティのプロパティであるプロパティ 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

しかし、その 2 番目のプロパティ アクセスを追加すると、次のようになります。

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.0ldarg.0set_Property

コードが機能しない理由は、IL がスタック ベースの言語であるためです。パラメーターのないインスタンス メソッド (プロパティ ゲッターなど) を呼び出すと、スタックの一番上にあるオブジェクト (ここでは の結果unbox.any) は次のようになります。として使用されますがthis、これはここで必要なものではありません。基本的に、あなたのコードは次のようなことをしようとします:

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