С# Использование ключевого слова Dynamic для доступа к свойствам через строки без отражения

Я хотел бы написать что-то похожее на следующее:

//  I will pass in a number of "properties" specified as strings that I want modified
string[] properties = new [] { "AllowEdit", "AllowDelete" };

//  Casting the component I'm using to a dynamic object of some sort ?
dynamic d = myGridComponent;

//  Iterate over the strings and set the properties
foreach(var s in properties) 
{
  //d.s = true; // 
  //d[s] = true; // this format would be ideal
}

Мне было интересно, если бы был простой способ сделать это, не используя Reflection [ .GetProperty(...).GetValue(...,...)], используя новое ключевое слово С# 4.0: dynamic.

Кажется, что может быть какой-то способ... Я просто не уверен в точном механизме и не смог найти нужный ресурс, чтобы собрать все части.

Мысли?

[EDIT] Похоже, есть пакет под названием "Clay", который каким-то образом реализует этот тип функций. Clay on CodePlex
Скотт Гензельман по теме

Ответы

Ответ 1

Это можно сделать. Вам просто нужно переопределить TryGetIndex на DynamicObject. Мне нужно что-то похожее на вызов статических членов типа, но, надеюсь, вы получите эту идею. Имейте в виду, что это в настоящее время не работает над методами с аргументами или типами обобщенного типа, которые перегружены, ограничивая его полезность:

internal class StaticMembersDynamicWrapper : DynamicObject
{
    private readonly IDictionary<String, MemberInfo> staticMembers = new Dictionary<string, MemberInfo>();
    private readonly Type type;

    public StaticMembersDynamicWrapper(Type type)
    {
        this.type = type;
        type.GetMembers(BindingFlags.FlattenHierarchy | BindingFlags.Static | BindingFlags.Public)
            .Each(member => staticMembers[member.Name] = member);
    }

    public override bool TryGetIndex(GetIndexBinder binder, object[] indexes, out object result)
    {
        var name = indexes[0] as string;

        MemberInfo member;

        if (false == staticMembers.TryGetValue(name, out member))
        {
            result = null;
            return false;
        }

        var prop = member as PropertyInfo;
        if (prop != null)
        {
            result = prop.GetValue(null, null);
            return true;
        }
        var method = member as MethodInfo;
        if (method != null)
        {
            var parameterTypes = (from p in method.GetParameters()
                                  select p.ParameterType).ToArray();
            var delegateType = method.ReturnType != typeof (void)
                            ? Expression.GetFuncType(parameterTypes.Union(new[]{method.ReturnType}).ToArray())
                            : Expression.GetActionType(parameterTypes);
            result = Delegate.CreateDelegate(delegateType, method);
            return true;
        }
        result = null;
        return false;
    }
}

dynamic d = new StaticMembersDynamicWrapper(typeof(string));
var result = d["IsNullOrEmpty"](String.Empty);

Ответ 2

Нет. dynamic в С# этого не предлагает. С двумя примерами:

d.s = true; // this looks for a property or field called "s"
d[s] = true; // this looks for an indexer that matches the string/bool signature

Вы можете написать тот же код, что и dynamic, но это было бы намного сложнее, чем просто использовать отражение. Либо используйте отражение (в соответствии с вашим примером), либо если вам нужно его оптимизировать, вы можете его обернуть в делегат через Expression или Delegate.CreateDelegate.

Ответ 3

Рамка openource Impromptu-interface, доступная через nuget, инкапсулирует код, который динамический будет генерировать полностью динамически. Это не так быстро, как использование динамического ключевого слова, но быстрее, чем отражение.

foreach(var s in properties) 
{
  //d.s = true; 
  Impromptu.InvokeSet(d, s, true);
}