Crear predicado con clases anidadas con expresión

Tengo esto :

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

Me gustaría que un caso genere el predicado de esta manera:

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

O esto :

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

O esto :

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

O esto :

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

Luego creé un “PredicateBuilder”, que es un trabajo (obtengo el tipo, si es nullable o no y construyo el predicado) cuando hago esto:

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

Pero no sé cómo administrar cuando hay una propiedad anidada como esta:

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

O esto :

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

O esto :

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

(sin tener la intención de duplicar a Jon – OP se contactó conmigo para proporcionar una respuesta)

Lo siguiente parece funcionar bien:

 static Expression> BuildPredicate(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>(Expression.Equal( body, Expression.Constant(value, body.Type)), p); } 

La única diferencia funcional entre eso y la respuesta de Jon es que se maneja null poco mejor, al decirle a Expression.Constant cuál es el tipo esperado. Como demostración de uso:

 static void Main() { var pred = BuildPredicate("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); } 

Solo necesita dividir su expresión por puntos y luego iterar sobre ella, usando Expression.Property varias veces. Algo como esto:

 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>(predicate, parameter);