采用表达式树(Expression Tree)对一个对象的属性进行“遍历”
时间:2014-02-27 11:41:36
收藏:0
阅读:515
除去直接对类进行访问的方式之外,目前已经有三种方式,可以读取一个未知类型的对象的属性或字段。第一种也就是最常见的反射了,实现起来较为简单,但是如果每次要访问同一个类型的大量对象,则性能很差。第二种是采用Delegate的方式,参见:《采用Delegate对一个对象进行遍历,http://blog.csdn.net/kmguo/article/details/17392185》 这种方式也有缺点,就是无法读取一个非基本类型及(String)类型的属性或字段。示例:
执行结果:

public class Student
{
public int Id { get; set; }
public string Name { get; set; }
public Location Location { get; set; }
}
public class Location
{
public int Row { get; set; }
public int Col { get; set; }
}采用Delegate的方式,只能获得Student的实例对象的Id及Name值(与String类的特殊性相关,因为调用它的ToString()方法就能直接得到我们想要的字符串),而不能得到Location里的Row及Col值(因为Location实例的ToString方法默认不会返回我们想要的Row及Col属性值)。
但是,可以采用Expression树的方式来得到,先建立表达式树,最后再将表达式树编译成Delegate来访问。
假设Student的一个实象对象为:student,在一般情况下,我们对其所有的公有属性,主要有以下的访问方式:
属性名称 平时的访问方式 Delegate的访问方式(Func<Student, object>)
Id student.Id s=>s.Id
Name student.Name s=>s.Name
Location.Row student.Location.Row s=>s.Location.Row
Location.Col student.Location.Col s=>s.Location.Col
现在,我们可以采用Expression Tree的方式,“模拟”Delegate的访问方式,并最终“编译”成Delegate,即 Func<Tin,Tout>。为进一步说明采用Expression Tree的功能,现假设一个场景:我们要获得一个未知类型的对象的所有公有属性的值,如果属性不是基本类型或String,就进一步获得其内部的公有属性的值。以上面的Student对象为例.
using System;
using System.Linq.Expressions;
using System.Reflection;
namespace ConsoleApplication1
{
class Program
{
static void Main(string[] args)
{
object student = new Student
{
Id = 1,
Name = "zhang san",
Location = new Location
{
Row = 10,
Col = 20
}
};
VisitProperties<Student>(student);
}
/// <summary>
/// 对未知类型的对象的属性进行递归访问
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="obj"></param>
static void VisitProperties<T>(object obj)
{
var type = obj.GetType();
var paraExpression = Expression.Parameter(typeof(T), "object");
foreach (var prop in type.GetProperties())
{
var propType = prop.PropertyType;
//判断是否为基本类型或String
//访问方式的表达式树为:obj =>obj.Property
if (propType.IsPrimitive || propType == typeof (String))
{
VisitProperty<T>(obj, prop, paraExpression, paraExpression);
}
else
{
//对于访问方式的表达式树为: obj=>obj.otherObj.Property。
Console.WriteLine("not primitive property: " + prop.Name);
var otherType = prop.PropertyType;
MemberExpression memberExpression = Expression.Property(paraExpression, prop);
//访问obj.otherObj里的所有公有属性
foreach (var otherProp in otherType.GetProperties())
{
VisitProperty<T>(obj, otherProp, memberExpression, paraExpression);
}
}
Console.WriteLine("--------------------------------");
}
}
/// <summary>
/// 执行表达式树为: obj=>obj.Property 或 obj=>obj.otherObj.Property的计算
/// </summary>
/// <param name="instanceExpression">最终访问属性的obj对象的表达式树的表示</param>
/// <param name="parameterExpression">类型T的参数表达式树的表示</param>
static void VisitProperty<T>(Object obj, PropertyInfo prop, Expression instanceExpression, ParameterExpression parameterExpression)
{
Console.WriteLine("property name: " + prop.Name);
MemberExpression memExpression = Expression.Property(instanceExpression, prop);
//实现类型转换,如将Id的int类型转为object类型,便于下面的通用性
Expression objectExpression = Expression.Convert(memExpression, typeof(object));
Expression<Func<T, object>> lambdaExpression = Expression.Lambda<Func<T, object>>(objectExpression, parameterExpression);
//打印表达式树
Console.WriteLine("expression tree: " + lambdaExpression);
Func<T, object> func = lambdaExpression.Compile();
Console.WriteLine("value: " + func((T)obj)); //打印出得到的属性值
}
}
}
执行结果:
采用表达式树(Expression Tree)对一个对象的属性进行“遍历”,布布扣,bubuko.com
原文:http://blog.csdn.net/kmguo/article/details/19975331
评论(0)