表达式树是一个不错的功能,但是它的实际用途是什么?它们可以用于某种形式的代码生成或元编程吗?
正如Jon所述,我使用它们为generic operators提供.NET 3.5。我还使用它们(再次在MiscUtil中)提供对非默认构造函数的快速访问(您不能将Delegate.CreateDelegate
与构造函数一起使用,但Expression
可以正常工作)。
手动创建的表达式树的其他用途:
但是实际上,Expression是编写任何动态代码的一种非常通用的方法。比Reflection.Emit
简单得多,而且对我来说,比CodeDOM更容易理解。在.NET 4.0中,您可以使用even more options。我展示了通过Expression
on my blog编写代码的基础。
Marc Gravell在MiscUtil中使用它们发挥了很大的作用,以实现generic operators。
[我刚刚使用要与大家共享的表达式树创建了通用过滤器功能...
开始
var allFiltered= Filter(AllCustomer, "Name", "Moumit");
public static List<T> Filter<T>(this List<T> Filterable, string PropertyName, object ParameterValue)
{
ConstantExpression c = Expression.Constant(ParameterValue);
ParameterExpression p = Expression.Parameter(typeof(T), "xx");
MemberExpression m = Expression.PropertyOrField(p, PropertyName);
var Lambda = Expression.Lambda<Func<T, Boolean>>(Expression.Equal(c, m), new[] { p });
Func<T, Boolean> func = Lambda.Compile();
return Filterable.Where(func).ToList();
}
一个更多
string singlePropertyName=GetPropertyName((Property.Customer p) => p.Name);
public static string GetPropertyName<T, U>(Expression<Func<T, U>> expression)
{
MemberExpression body = expression.Body as MemberExpression;
// if expression is not a member expression
if (body == null)
{
UnaryExpression ubody = (UnaryExpression)expression.Body;
body = ubody.Operand as MemberExpression;
}
return string.Join(".", body.ToString().Split('.').Skip(1));
}
使其更具扩展性
string multiCommaSeparatedPropertyNames=GetMultiplePropertyName<Property.Customer>(c => c.CustomerId, c => c.AuthorizationStatus)
public static string GetMultiplePropertyName<T>(params Expression<Func<T, object>>[] expressions)
{
string[] propertyNames = new string[expressions.Count()];
for (int i = 0; i < propertyNames.Length; i++)
{
propertyNames[i] = GetPropertyName(expressions[i]);
}
return propertyNames.Join();
}
我知道也可以使用反射来完成,但是这非常快,或者,我可以说,相当于第一次编译后的lambda。第一次迭代平均只需要10毫秒。这就是表情树魔术。简单而梦幻(我认为)!
我使用它们来创建动态查询,无论是用于排序还是过滤数据。例如:
IQueryable<Data.Task> query = ctx.DataContext.Tasks;
if (criteria.ProjectId != Guid.Empty)
query = query.Where(row => row.ProjectId == criteria.ProjectId);
if (criteria.Status != TaskStatus.NotSet)
query = query.Where(row => row.Status == (int)criteria.Status);
if (criteria.DueDate.DateFrom != DateTime.MinValue)
query = query.Where(row => row.DueDate >= criteria.DueDate.DateFrom);
if (criteria.DueDate.DateTo != DateTime.MaxValue)
query = query.Where(row => row.DueDate <= criteria.DueDate.DateTo);
if (criteria.OpenDate.DateFrom != DateTime.MinValue)
query = query.Where(row => row.OpenDate >= criteria.OpenDate.DateFrom);
var data = query.Select(row => TaskInfo.FetchTaskInfo(row));
LINQ提供程序的实现主要是通过处理表达式树来完成的。我还使用它们从代码中删除文字字符串:
我使用表达式树构建数学表达式评估器:Building Expression Evaluator with Expression Trees in C#
您可以使用它们为Google或Flickr或Amazon之类的网站,您自己的网站或其他数据提供商构建自己的linq提供程序。
Jomo Fisher最初发布,Gustavo Guerra发布了static string dictionary。
通过表达式树,这是一个动态表达式,提供了一个真正的(荒谬的)字典。
该实现创建一个动态决策树,该树根据输入字符串的长度,然后按照第一个字母,然后是第二个字母,依此类推,选择相应的值。
这最终比等效的Dictionary运行得快得多。