使用參數的索引器構建表達式樹

c# expression expression-trees lambda linq

給定一個具有Dictionary屬性的類

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

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

我希望能夠根據從數據存儲中檢索到的格式為的條件匹配產品列表中的產品

Brand == Tyco
Color != Blue

我目前的方法是從過濾器構造一個表達式,然後將該表達式作為參數傳遞給LINQ Where方法調用

products = products.Where(myConstructedExpression);

其中myConstructedExpression通常是一個看起來像的lamda表達式

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

為了測試的目的,我已經彙編了以下代碼,但它總是無法調用lambda.Compile()無論我為他留下的表達式做了什麼。

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);

問題

  1. 這甚至是一種合理的方法,如果是這樣的話,
  2. 如何構造左表達式?

一般承認的答案

所以要獲得p => p.Attributes["Brand"] <someoperator> "Tyco" ,你可以這樣做。

使用索引類型的“技巧”是使用它們的Item屬性(您也可以使用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) });

編輯

IDictionary.ContainsKey(<value>)測試的版本

真的是一步一步,但我認為這首先讓事情變得更加清晰。

//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);


許可下: CC-BY-SA with attribution
不隸屬於 Stack Overflow
這個KB合法嗎? 是的,了解原因
許可下: CC-BY-SA with attribution
不隸屬於 Stack Overflow
這個KB合法嗎? 是的,了解原因