Expression trees represent code in a tree-like data structure, where each node is an expression, for example, a method call or a binary operation

MSDN的原文已经说明表达式树的作用就是以树状的数据结构来表示代码,树中的每个节点就是一个表达式,如方法调用或者二进制操作.表达式树的作用就是把代码以数据的形式来表达,
方便修改编辑,然后再转换为相应的代码来执行.

如何创建表达式树

我们在用委托这种类型来表示.net中的执行代码(方法) , 而做为代码的数据表示形式的表达式树我们是使用System.Linq.Expressions命名空间的Expression类型来表示.
例如委托Func<int, int>表示参数是int,返回值是int的实例方法或者静态方法的委托类型,那么Expression<Func<int, int>> 就是表示构建此类委托的数据结构,用于动态构建我们的方法.

  1. 直接使用lamda表达式来创建.
    static void Main(string[] args)
            {
                Expression<Func<int, int>> expresstion = x =>   x + 5;
                var expressionCompiled=expresstion.Compile();
                int input = 5;
                Console.WriteLine("input:{0},output:{1}",input, expressionCompiled(input));
                Console.Read();
            }



    使用lamda表达式来创建表达式树有限制,就是lamda表达式只能是语句而不能是表达式块.
    例如:

    Expression<Func<int, int>> expresstion = x => { return x + 5; };

    编译器会报错:

  2. 使用Expression类封装好的方法来创建表达式树.
                //construct the param expression
                var paramExpression= Expression.Parameter(typeof (int), "x");
                //and then the body expression
                var constantExpression=Expression.Constant(5, typeof (int));
                var bodyExpression= Expression.MakeBinary(ExpressionType.Add, paramExpression, constantExpression);
                //use the Lamda() method to construct the expression tree
                var func= Expression.Lambda<Func<int, int>>(bodyExpression, paramExpression).Compile();
                int input = 5;
                Console.WriteLine("input:{0},output:{1}",input, func(5));
    

执行代码

表达式树只是代码的数据形式,不是执行代码,并不能直接执行.要转换为代码,需要调用Compile()方法编译,并返回委托对象.

实验一下:值对象的相等比较

DDD的值对象没有唯一的身份标识,直接来说就是没有唯一的ID(或者应该说即使ID不同的对象也可能看做是相同的对象,判断他们是否相同只能比较他们具体的一些属性,值类型不会单独存在,它应该作为实体类型或者其他值类型的属性依附存在.)
例如表示位置的值类型:

 public class Position
    {
        /// <summary>
        /// 经度
        /// </summary>
        [EqualKey]
        public decimal Lon { get; set; }
        /// <summary>
        /// 纬度
        /// </summary>
        [EqualKey]
        public decimal Lat { get; set; }
        /// <summary>
        /// 文字描述
        /// </summary>
        public string Description { get; set; }
    }

我们只需要根据经度跟纬度这两个属性是否相等来判断Positon值是否相等,经度跟纬度这两个关键的属性会使用自定义的特征EqualKeyAttribute来标识.
所有的值类型都继承自类型抽象基类ValueObject,且基类的Equal方法被重写,相等比较的逻辑在虚方法中通过反射寻找每个值类型的关键属性来构建表达式树,
最后编译成特定的Func委托实现.

/// <summary>
    /// 值类型基类
    /// </summary>
    public abstract class ValueObject<T> where T:ValueObject<T>
    {

        public override bool Equals(object obj)
        {
            if (obj == null) return false;
            if (this.GetType() != obj.GetType()) return false;
            return BuildValueComparer()((T)this, (T)obj);
        }

        /// <summary>
        /// 建立值比较委托(子类可重写)
        /// </summary>
        /// <returns></returns>
        protected virtual Func<T, T, bool> BuildValueComparer()
        {
            return ValueObjectComparerFactory.BuildComparer<T>();
        }

        /// <summary>
        /// 重载==
        /// </summary>
        /// <returns></returns>
        public static bool operator ==(ValueObject<T> source,ValueObject<T> compare)
        {
            if (source == null)
            {
                return compare == null;
            }
            return source.Equals(compare);
        }
        /// <summary>
        /// 重载!=
        /// </summary>
        /// <returns></returns>
        public static bool operator !=(ValueObject<T> source, ValueObject<T> compare)
        {
            return !(source == compare);
        }

    }

    [AttributeUsage(AttributeTargets.Property)]
    public class EqualKeyAttribute : Attribute
    {

    }

    internal interface IValueObjectComparerBuilder<T>
    {
        Func<T, T, bool> BuildComparer<T>();
    }

    static class ValueObjectComparerFactory
    {
        private static Dictionary<Type, object> cache = new Dictionary<Type, object>();
        private static object syncObject = new object();
        internal static Func<T, T, bool> BuildComparer<T>()
        {
            var objectType = typeof (T);
            if (!cache.ContainsKey(objectType))
            {
                lock (syncObject)
                {
                    if (!cache.ContainsKey(objectType))
                    {
                        IValueObjectComparerBuilder<T> builder = new ExpressionTreeComparerBuilder<T>();
                        cache.Add(objectType, builder.BuildComparer<T>());
                    }
                }
            }
            return cache[objectType] as Func<T, T, bool>;
        }
    }

    internal class ExpressionTreeComparerBuilder <T>: IValueObjectComparerBuilder<T>
    {
        public Func<T, T, bool> BuildComparer<T>()
        {
            Expression<Func<T, T, bool>> expression = null;
            Expression finalExpression = null;
            Type objectType = typeof (T);
            //parameter a,b
            var paramA = Expression.Parameter(objectType, "a");
            var paramB = Expression.Parameter(objectType, "b");
            //var typeParamA = Expression.Convert(paramA, objectType);
            //var typeParamB = Expression.Convert(paramB, objectType);

            foreach (var propertyInfo in objectType.GetProperties())
            {
                if (propertyInfo.GetCustomAttribute<EqualKeyAttribute>(false) != null)
                {
                    var propertyExpressionA = Expression.Property(paramA, propertyInfo);
                    var propertyExpressionB = Expression.Property(paramB, propertyInfo);
                    var equalExpression = Expression.MakeBinary(ExpressionType.Equal, propertyExpressionA, propertyExpressionB);
                    if (finalExpression == null)
                        finalExpression = equalExpression;
                    else
                        finalExpression = Expression.MakeBinary(ExpressionType.AndAlso, finalExpression, equalExpression);
                }
            }
            if (finalExpression != null)
                return Expression.Lambda<Func<T, T, bool>>(finalExpression, paramA, paramB).Compile();
            else
                return (a, b) => a.Equals(b);
        }
    }

测试一下:

static void Main(string[] args)
        {
            Position pos1 = new Position
            {
                Lat = 100,
                Lon = 100,
                Description = "佛山市"
            };

            Position pos2 = new Position
            {
                Lat = 100,
                Lon = 100,
                Description = "顺德区"
            };

            Position pos3 = new Position
            {
                Lat = 111,
                Lon = 111,
                Description = "佛山市"
            };

            Console.WriteLine("{0}=={1} Result=>{2}",pos1.ToString(),pos2.ToString(), pos1.Equals(pos2).ToString());
            Console.WriteLine("{0}=={1} Result=>{2}", pos1.ToString(), pos3.ToString(), pos1.Equals(pos3).ToString());
            Console.Read();
        

To be continued………………


参考:
https://blogs.msdn.microsoft.com/charlie/2008/01/31/expression-tree-basics/

https://docs.microsoft.com/en-us/dotnet/csharp/programming-guide/concepts/expression-trees/index

LINQ Expresstion Tree 表达式树的更多相关文章

  1. Linq系列(7)——表达式树之ExpressionVisitor

    大家好,由于今天项目升级,大家都在获最新代码,所以我又有时间在这里写点东西,跟大家分享. 在上一篇的文章中我介绍了一个dll,使大家在debug的时候可以可视化的看到ExpressionTree的Bo ...

  2. Linq系列(5)——表达式树之案例应用

    在进入今天的正题之前,先感慨下本人的blog的人气一篇不如一篇.再加上换公司后人身自由受到了比之前大得多得多的限制,实在令本人有些郁闷.不过每次提笔写些东西跟大家分享,总是能让我感到愉悦和欣慰,希望我 ...

  3. C# - LINQ 表达式树

    表达式树(Expression Tree) 表达式树是不可执行的代码,它只是用于表示一种树状的数据结构,树上的每一个节点都表示为某种表达式类型,大概有25种表达式类型,它们都派生自Expression ...

  4. 表达式树(Expression Tree)

    你每创建一个表示表达式的实例时,都可以将该类型实例看成是一棵表达式树.每种表示表达式的类型都有一个具体的类型,如Expression的Variable()方法创建的是ParameterExpressi ...

  5. 追根溯源之Linq与表达式树

    一.什么是表达式树?   首先来看下官方定义(以下摘录自巨硬官方文档)   表达式树表示树状数据结构中的代码,其中每个节点都是表达式,例如,方法调用或诸如的二进制操作x < y.   您可以编译 ...

  6. [.net 面向对象程序设计进阶] (6) Lamda表达式(二) 表达式树快速入门

    [.net 面向对象程序设计进阶] (6) Lamda表达式(二) 表达式树快速入门 本节导读: 认识表达式树(Expression Tree),学习使用Lambda创建表达式树,解析表达式树. 学习 ...

  7. 转载:C#特性-表达式树

    原文地址:http://www.cnblogs.com/tianfan/ 表达式树基础 刚接触LINQ的人往往觉得表达式树很不容易理解.通过这篇文章我希望大家看到它其实并不像想象中那么难.您只要有普通 ...

  8. C#特性-表达式树

    表达式树ExpressionTree   表达式树基础 转载需注明出处:http://www.cnblogs.com/tianfan/ 刚接触LINQ的人往往觉得表达式树很不容易理解.通过这篇文章我希 ...

  9. C#编程(六十六)----------表达式树总结

    表达式树总结 基础 表达式树提供了一个将可执行代码转换成数据的方法.如果你要在执行代码之前修改或转换此代码,那么它是很有用的.有其是当你要将C#代码----如LINQ查询表达式转换成其他代码在另一个程 ...

随机推荐

  1. Oracle update语句更新值来自另一张表中的数据

    task 任务表 role 角色表 两表之间必须有关联的字段 update task t set t.roleName = ( select r.name from role r where r.id ...

  2. NetSec2019 20165327 Exp1 PC平台逆向破解

    NetSec2019 20165327 Exp1 PC平台逆向破解 一.实践目标 本次实践的对象是一个名为pwn1(实验中用的 是5327pwnx)的linux可执行文件. 该程序正常执行流程是:ma ...

  3. 自动生成CHANGELOG.md

    $ npm install -g conventional-changelog-cli $ cd my-project $ conventional-changelog -p angular -i C ...

  4. Linux删除/boot后该如何恢复

    一.其实不光boot目录被删除,也有系统启动不起来等,一直因为/boot这个目录的的问题,今天就来解决一下 Centos 6 1.进入救援,先修复vmlinz(内核) chroot /mnt/sysi ...

  5. h5页面避免两个页面反复跳转死循环

    项目中经常会碰到在一个页面加判断之后跳转另一个页面,并且无法返回,来回跳转,死循环,遇到这种情况可以在跳转时使用location,replace,关闭之前页面,与location.href 两者的不同 ...

  6. 一些重要的k8s概念

    1. Pause容器是什么 作为init pod存在,其他pod都会从pause 容器中fork出来 由pause容器管理 pause容器的工作 可知kubernetes的pod抽象基于Linux的n ...

  7. [bzoj P2726] [SDOI2012]任务安排

    [bzoj P2726] [SDOI2012]任务安排 Time Limit: 10 Sec Memory Limit: 128 MB Submit: 1204 Solved: 349[Submit] ...

  8. 不同浏览器中scrollHeight的比较

    每个浏览器的兼容性不同所以就有了如下判断方案 document.body是DOM中Document对象里的body节点, document.documentElement是文档对象根节点(html)的 ...

  9. Azkaban启动web--javax.net.ssl.SSLException: Unrecognized SSL message, plaintext connection? at sun.se

    javax.net.ssl.SSLException: Unrecognized SSL message, plaintext connection? at sun.se javax.net.ssl. ...

  10. java操作svn工具类SvnUtil

    直接上代码,工作中使用的版本,记录下. public class SvnUtil { private static Logger logger = Logger.getLogger(SvnUtil.c ...