C#IL代码将值添加到Dictionary

时间:2011-05-12 00:19:59

标签: c# il

我有一个动态方法,它采用DataRecord并将数据映射到某种类型的对象。

来自DynamicMethod_ILGenerator

对象有一个类型为Dictionary<string,object>的字典,用于保存DataRecord中无法绑定的任何属性。

但是我似乎无法弄清楚如何将这些添加到词典中。

我知道我必须在字典上拨打Emit(OpCodes.Callvirt, addMethod),但无论我尝试什么,我都无法弄清楚如何让它发挥作用。

public static Load CreateBuilder(IDataRecord dataRecord)
    {
        var dynamicBuilder = new DynamicBuilder<T>();

        var method = new DynamicMethod("DynamicCreate", typeof(T), new[] { typeof(IDataRecord) }, typeof(T), true);
        var generator = method.GetILGenerator();

        var result = generator.DeclareLocal(typeof(T));
        generator.Emit(OpCodes.Newobj, typeof(T).GetConstructor(Type.EmptyTypes));
        generator.Emit(OpCodes.Stloc, result);


        // Dictionary to store.
        var dictType = typeof (Dictionary<string, object>);
        var dict = typeof(T).GetProperty("CustomFields");
        var addMethod = dictType.GetMethod("Add");

        for (var i = 0; i < dataRecord.FieldCount; i++)
        {
            var propertyInfo = typeof(T).GetProperty(dataRecord.GetName(i));
            var endIfLabel = generator.DefineLabel();

            if (propertyInfo != null && propertyInfo.GetSetMethod() != null)
            {
                var attributes = propertyInfo.GetCustomAttributes(typeof(DbIgnore), true) as DbIgnore[];
                if (attributes != null && attributes.Length > 0) continue;

                bool isNullable = false;
                if (propertyInfo.PropertyType.Name.ToLower().Contains("nullable"))
                    isNullable = true;

                Type _type = dataRecord.GetFieldType(i);

                generator.Emit(OpCodes.Ldarg_0);
                generator.Emit(OpCodes.Ldc_I4, i);
                generator.Emit(OpCodes.Callvirt, IsDbNullMethod);
                generator.Emit(OpCodes.Brtrue, endIfLabel);

                generator.Emit(OpCodes.Ldloc, result);
                generator.Emit(OpCodes.Ldarg_0);
                generator.Emit(OpCodes.Ldc_I4, i);
                generator.Emit(OpCodes.Callvirt, GetValueMethod);

                if (isNullable)
                    generator.Emit(OpCodes.Unbox_Any, GetNullableType(_type));
                else
                    generator.Emit(OpCodes.Unbox_Any, _type);

                //generator.Emit(OpCodes.Unbox_Any, dataRecord.GetFieldType(i));
                generator.Emit(OpCodes.Callvirt, propertyInfo.GetSetMethod());

                generator.MarkLabel(endIfLabel);
            } else {
                  // How-To Add to Dictionary
            }
        }

        generator.Emit(OpCodes.Ldloc, result);
        generator.Emit(OpCodes.Ret);

        return (Load)method.CreateDelegate(typeof(Load));
    }

1 个答案:

答案 0 :(得分:1)

在循环外部,将对象的Dictionary引用提取到局部变量中。

在循环内部,将包含字典变量,键和值的局部变量推送到堆栈,然后将callvirt发送到Add(TKey, TValue)方法。