Utiliser la valeur de retour lambda pour l'arbre d'expression

c# expression-trees lambda

Question

J'essaie de jouer un peu avec les arbres d'expression. J'ai un objet avec List<string> et je veux construire un arbre d'expression qui ajoute une valeur à cette propriété, mais je veux spécifier la valeur à ajouter via un Func . Actuellement j'essaye ça ...

public static Action<T> CreateMethodAddObjectToList<T, C>(this Type type, string property, Func<C> ctorFunction)
        {
            PropertyInfo fieldInfo = type.GetProperty(property);

            if (fieldInfo == null)
            {
                return null;
            }

            ParameterExpression targetExp = Expression.Parameter(type, "target");
            MemberExpression fieldExp = Expression.Property(targetExp, property);
            var method = fieldExp.Type.GetMethod("Add", BindingFlags.Public | BindingFlags.Instance);

            Expression<Func<C>> ctorExpression = () => ctorFunction();

// but this doesnt work because I can't use the ctorExpression in this way
            var callExp = Expression.Call(fieldExp, method, ctorExpression);

            var function = Expression.Lambda<Action<T>>(callExp, targetExp).Compile();

            return function;
        }

l'appel ressemble à

var dummyObject = new DummyObject { IntProperty = 5 };

            Action<DummyObject> setter = typeof (DummyObject).CreateMethodAddObjectToList<DummyObject, string>("StringList", () => "Test" );

Réponse acceptée

Vous pouvez remplacer ctorFunction par une Expression<Func<C>> , puis l'invoquer dans l'action générée:

public static Action<T> CreateMethodAddObjectToList<T, C>(this Type type, string property, Expression<Func<C>> createExpr)
{
    PropertyInfo fieldInfo = type.GetProperty(property);

    if (fieldInfo == null)
    {
        return null;
    }

    ParameterExpression targetExp = Expression.Parameter(type, "target");
    MemberExpression fieldExp = Expression.Property(targetExp, property);
    var method = fieldExp.Type.GetMethod("Add", BindingFlags.Public | BindingFlags.Instance);

    var valueExpr = Expression.Invoke(createExpr);
    var callExpr = Expression.Call(fieldExp, method, valueExpr);

    var function = Expression.Lambda<Action<T>>(callExpr, targetExp).Compile();

    return function;
}



Sous licence: CC-BY-SA with attribution
Non affilié à Stack Overflow
Est-ce KB légal? Oui, apprenez pourquoi
Sous licence: CC-BY-SA with attribution
Non affilié à Stack Overflow
Est-ce KB légal? Oui, apprenez pourquoi