使用Expression创建具有嵌套类的谓词

时间:2013-01-10 13:20:07

标签: c# .net linq expression predicate

我有这个:

public class Company
{
    public int Id { get; set; }
    public string Name { get; set; }
}
public class City
{
    public int Id { get; set; }
    public string Name { get; set; }
    public int ZipCode { get; set; }
}
public class Person
{
    public int Id { get; set; }
    public string FirstName { get; set; }
    public string LastName { get; set; }
    public int? Age { get; set; }
    public City City { get; set; }
    public Company Company { get; set; }
}

我想在某种情况下生成这样的谓词:

var result = listPerson.Where(x => x.Age == 10).ToList<>();

或者这个:

var result  = listPerson.Where( x => x.Company.Name == 1234).ToList();

或者这个:

var result  = listPerson.Where( x => x.City.ZipCode == "MyZipCode").ToList();

或者这个:

var result  = listPerson.Where( x => x.Company.Name == "MyCompanyName").ToList();

然后我创建了一个“PredicateBuilder”,这是我的工作(我得到类型,如果可以为空,我构建谓词)当我这样做时:

BuildPredicate<Person>("Age", 10); I get this : x => x.Age == 10

但是当我有这样的嵌套属性时,我不会如何管理:

BuildPredicate<Person>("City.ZipCode", "MyZipCode"); 
I'd like get this : x => x.City.ZipCode == "MyZipCode"

或者这个:

BuildPredicate<Person>("City.Name", "MyName"); 
I'd like get this : x => x.City.Name == "MyName"

或者这个:

BuildPredicate<Person>("Company.Name", "MyCompanyName"); 
I'd like get this : x => x.Company.Name == "MyCompanyName"

2 个答案:

答案 0 :(得分:9)

(不打算复制Jon - OP联系我提供答案)

以下似乎工作正常:

static Expression<Func<T,bool>> BuildPredicate<T>(string member, object value) {
    var p = Expression.Parameter(typeof(T));
    Expression body = p;
    foreach (var subMember in member.Split('.')) {
        body = Expression.PropertyOrField(body, subMember);
    }
    return Expression.Lambda<Func<T, bool>>(Expression.Equal(
        body, Expression.Constant(value, body.Type)), p);
}

与Jon的答案之间唯一的功能区别在于它通过告诉null期望的类型是什么来更好地处理Expression.Constant 稍微。作为使用示范:

static void Main() {
    var pred = BuildPredicate<Person>("City.Name", "MyCity");

    var people = new[] {
        new Person { City = new City { Name = "Somewhere Else"} },
        new Person { City = new City { Name = "MyCity"} },
    };
    var person = people.AsQueryable().Single(pred);
}

答案 1 :(得分:5)

您只需要按点分割表达式,然后使用Expression.Property多次迭代它。像这样:

string[] properties = path.Split('.');
var parameter = Expression.Parameter(typeof(T), "x");
var lhs = parameter;
foreach (var property in properties)
{
    lhs = Expression.Property(lhs, property);
}
// I've assumed that the target is a string, given the question. If that's
// not the case, look at Marc's answer.
var rhs  = Expression.Constant(targetValue, typeof(string));
var predicate = Expression.Equals(lhs, rhs);
var lambda = Expression.Lambda<Func<T, bool>>(predicate, parameter);