Construction d'un arbre d'expression à l'aide d'un indexeur de paramètre

c# expression expression-trees lambda linq

Question

Étant donné une classe qui a une propriété qui est un dictionnaire

public class Product
{
    public Dictionary<string, string> Attributes { get { return attributes; } }

    private Dictionary<string, string> attributes = new Dictionary<string, string>();
}

Je souhaite pouvoir faire correspondre les produits dans une liste de produits en fonction de critères extraits d'un magasin de données et se présentant sous le format suivant:

Brand == Tyco
Color != Blue

Mon approche actuelle consiste à construire une expression à partir du filtre, puis à transmettre cette expression en tant que paramètre à un appel de méthode LINQ Where comme suit

products = products.Where(myConstructedExpression);

myConstructedExpression serait normalement une expression lamda qui ressemble à

p => p.Attributes[attribute] == value

J'ai assemblé le code suivant à des fins de test, mais l'appel à lambda.Compile() échoue toujours indépendamment de ce que j'ai essayé car il a quitté expression.

Dictionary<string, ExpressionType> expressionType = new Dictionary<string, ExpressionType>();
expressionType.Add("==", ExpressionType.Equal);
expressionType.Add("!=", ExpressionType.NotEqual);

string filter = "Brand == Tyco";
string[] fields = filter.Split(' ');
string attribute = fields[0];
string op = fields[1];
string value = fields[2];

Product product = new Product();
product.Attributes.Add("Brand", "Tyco"); 

var parameter = Expression.Parameter(typeof(Product), "p");
var left = /***** THIS IS WHAT I AM FAILING TO CONSTRUCT PROPERLY ********/
var right = Expression.Constant(value);
var operation = Expression.MakeBinary(expressionType[op], left, right);
var lambda = Expression.Lambda<Func<Product, bool>>(operation, parameter);

var result = lambda.Compile()(product);

Des questions

  1. Est-ce même une approche raisonnable et, si oui,
  2. Comment construire l'expression de gauche?

Réponse acceptée

Donc, pour obtenir p => p.Attributes["Brand"] <someoperator> "Tyco" , vous pouvez le faire.

Le "truc", pour travailler avec des types indexés, consiste à utiliser leur propriété Item (vous pouvez aussi travailler avec la méthode get_item )

var parameter = Expression.Parameter(typeof(Product), "p");
Expression left = Expression.Property(parameter, "Attributes");
left = Expression.Property(left, "Item", new Expression[] { Expression.Constant(attribute) });

MODIFIER

la version avec le test IDictionary.ContainsKey(<value>)

vraiment, étape par étape, mais je pense que cela rend les choses plus claires au début.

//left part of lambda, p
var parameter = Expression.Parameter(typeof(Product), "p");
//right part
//p.Attributes
Expression left = Expression.Property(parameter, "Attributes");

var method = typeof(IDictionary<string, string>).GetMethod("ContainsKey");
//p.Attributes.ContainsKey("Brand");
Expression containsExpression = Expression.Call(left, method, Expression.Constant(attribute));
//p.Attributes.Item["Brand"]
Expression keyExpression= Expression.Property(left, "Item", new Expression[] { Expression.Constant(attribute) });
//"Tyco"
var right = Expression.Constant(value);
//{p => IIF(p.Attributes.ContainsKey("Brand"), (p.Attributes.Item["Brand"] == "Tyco"), False)}
Expression operation = Expression.Condition(
                           containsExpression,
                           Expression.MakeBinary(expressionType[op], keyExpression, right), 
                           Expression.Constant(false));
var lambda = Expression.Lambda<Func<Product, bool>>(operation, parameter);



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