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 ответ

Решение

Вне вашей петли, возьмите Dictionary ссылка из вашего объекта в локальную переменную.

Внутри цикла поместите локальную переменную, содержащую словарную переменную, ключ и значение в стек, а затем выполните команду emit callvirt к Add(TKey, TValue) метод.

Другие вопросы по тегам