更新
感谢Marc帮助AlphaPagedList类is now available on CodePlex,如果有人感兴趣的话
原始
我正在尝试创建一个表达式树来返回以给定字符开头的元素。
IList<char> chars = new List<char>{'a','b'};
IQueryable<Dept>Depts.Where(x=> chars.Contains(x.DeptName[0]));
我想在任何IEnumerable上使用它,我提供lamdba给属性选择例如:
Depts.Alpha(x=>x.DeptName, chars);
我一直在尝试这个,但没有运气,任何帮助?
public static IQueryable<T> testing<T>(this IQueryable<T> queryableData, Expression<Func<T,string>> pi, IEnumerable<char> chars)
{
// Compose the expression tree that represents the parameter to the predicate.
ParameterExpression pe = Expression.Parameter(queryableData.ElementType, "x");
ConstantExpression ch = Expression.Constant(chars,typeof(IEnumerable<char>));
// ***** Where(x=>chars.Contains(x.pi[0])) *****
// pi is a string property
//Get the string property
Expression first = Expression.Constant(0);
//Get the first character of the string
Expression firstchar = Expression.ArrayIndex(pi.Body, first);
//Call "Contains" on chars with argument being right
Expression e = Expression.Call(ch, typeof(IEnumerable<char>).GetMethod("Contains", new Type[] { typeof(char) }),firstchar);
MethodCallExpression whereCallExpression = Expression.Call(
typeof(Queryable),
"Where",
new Type[] { queryableData.ElementType },
queryableData.Expression,
Expression.Lambda<Func<T, bool>>(e, new ParameterExpression[] { pe }));
// ***** End Where *****
return (queryableData.Provider.CreateQuery<T>(whereCallExpression));
}
答案 0 :(得分:2)
重新阅读问题之后( EDITED ) - 但请注意Expression.Invoke
在3.5SP1中不适用于EF(但它在LINQ-to-SQL中很好用) ):
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
class Dept
{
public string DeptName { get; set; }
}
public static class Program
{
static void Main()
{
IList<char> chars = new List<char>{'a','b'};
Dept[] depts = new[] { new Dept { DeptName = "alpha" }, new Dept { DeptName = "beta" }, new Dept { DeptName = "omega" } };
var count = testing(depts.AsQueryable(), dept => dept.DeptName, chars).Count();
}
public static IQueryable<T> testing<T>(this IQueryable<T> queryableData, Expression<Func<T,string>> pi, IEnumerable<char> chars)
{
var arg = Expression.Parameter(typeof(T), "x");
var prop = Expression.Invoke(pi, arg);
Expression body = null;
foreach(char c in chars) {
Expression thisFilter = Expression.Call(prop, "StartsWith", null, Expression.Constant(c.ToString()));
body = body == null ? thisFilter : Expression.OrElse(body, thisFilter);
}
var lambda = Expression.Lambda<Func<T, bool>>(body ?? Expression.Constant(false), arg);
return queryableData.Where(lambda);
}
}