通俗版《把事件当作对象进行传递》看不懂老赵的文章的朋友进

2022-05-09 11:09:54 浏览数 (1)

代码语言:javascript复制
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Linq.Expressions;
using System.Reflection;
namespace zhao1
{
    #region 基础
    public class Event<T>
    {
        private object m_instance;
        private MethodInfo m_addMethod;
        private MethodInfo m_removeMethod;
        public Event(Expression<Func<T>> eventExpr)
        {
            //eventExpr是LambdaExpression类型的实例,
            //他的Body属性是lambda表达式的主体,
            //这里把lambda表达式强制转换成表达式目录树的数据结构
            var memberExpr = eventExpr.Body as MemberExpression;
            //获取memberExpr包含对象的Expression 
            //如果不是null就将表达式目录树描述的lambda表达式编译为可执行代码。并返回一个委托
            //这个委托表示由 Expression<(Of <(TDelegate>)>)描述的lambda表达式
            this.m_instance = memberExpr.Expression == null ? null : Expression.Lambda<Func<object>>(memberExpr.Expression).Compile()();
            //指定控制绑定和由反射执行的成员和类型搜索方法的标志
            var bindingFlags = BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.InvokeMethod |
                (this.m_instance == null ? BindingFlags.Static : BindingFlags.Instance);
            //表示memberExpr的MemberInfo。
            //MemberInfo获取有关成员属性的信息并提供对成员元数据的访问。
            var member = memberExpr.Member;
            //MemberInfo.DeclaringType属性获取声明该成员的类
            //GetMethod使用指定绑定约束搜索指定方法。
            //这里还是有些不明白
            this.m_addMethod = member.DeclaringType.GetMethod("add_"   member.Name, bindingFlags);
            this.m_removeMethod = member.DeclaringType.GetMethod("remove_"   member.Name, bindingFlags);
        }
        public Event<T> AddHandler(T handler)
        {
            //Invoke使用指定的参数调用当前实例所表示的方法或构造函数。
            this.m_addMethod.Invoke(this.m_instance, new object[] { handler });
            return this;
        }
        public Event<T> RemoveHandler(T handler)
        {
            this.m_removeMethod.Invoke(this.m_instance, new object[] { handler });
            return this;
        }
        public static Event<T> operator  (Event<T> ev, T handler)
        {
            return ev.AddHandler(handler);
        }
        public static Event<T> operator -(Event<T> ev, T handler)
        {
            return ev.RemoveHandler(handler);
        }
    }
    #endregion
    #region 工厂
    /// <summary>
    /// EventFactory类的Create方法可以避免显式地提供T类型
    /// </summary>
    public static class EventFactory
    {
        public static Event<T> Create<T>(Expression<Func<T>> eventExpr)
        {
            return new Event<T>(eventExpr);
        }
    }
    #endregion
    class Program
    {
        public event EventHandler Submit;
        static void Main(string[] args)
        {
            Program p = new Program();
            var ev = EventFactory.Create(() => p.Submit);
            ev  = (sender, eventArgs) => Console.WriteLine(sender);
            p.Submit("Hello World", EventArgs.Empty);
            p.Submit("xland is using vs2010",EventArgs.Empty);
            Console.ReadKey();
        }
    }
}

表达式目录树是 lambda 表达式在内存中的数据表示形式。 它使 lambda 表达式的结构变得更加透明而明确。 在与表达式目录树中的数据进行交互时,其方式就像与任何其他数据结构交互时一样。 这种将表达式处理为数据结构的功能,使 API 能够收到可按自定义方式进行检查、转换和处理的用户代码格式。

Expression 类 提供一种基类,表示表达式目录树节点的类派生自该基类。它还包含用来创建各种节点类型的 static工厂方法。这是一个 abstract 类。

Expression.Lambda 方法 创建一个表示 lambda 表达式的表达式目录树。

Expression<(Of <(TDelegate>)>) 泛型类 以表达式目录树的形式将强类型 lambda 表达式表示为数据结构。 Expression<(Of <(TDelegate>)>) 的 NodeType 为 Lambda。

Expression.Lambda<(Of <(TDelegate>)>) 泛型方法 创建一个在编译时委托类型已知的 Expression<(Of <(TDelegate>)>)。 使用 Lambda<(Of <(TDelegate>)>)(Expression, IEnumerable<(Of <(ParameterExpression>)>)) 或 Lambda<(Of <(TDelegate>)>)(Expression, array []()[]) 方法可创建 Expression<(Of <(TDelegate>)>) 对象。

Expression<(Of <(TDelegate>)>).Compile 方法 将表达式目录树描述的 lambda 表达式编译为可执行代码。 返回一个 TDelegate 类型的委托,表示由 Expression<(Of <(TDelegate>)>) 描述的 lambda 表达式。

MemberExpression 表示访问字段或属性。继承自Expression类

MemberExpression.Expression 属性 获取字段或属性包含对象的Expression

MethodInfo 发现方法的属性并提供对方法元数据的访问。

LambdaExpression 类 描述一个 lambda 表达式。LambdaExpression 类型以表达式目录树的形式表示 lambda 表达式。Expression<(Of <(TDelegate>)>) 类型派生自 LambdaExpression,它能够更加显式地捕获 lambda 表达式的类型,也可以用于表示 lambda 表达式。在运行时,表示 lambda 表达式的表达式目录树节点始终属于类型 Expression<(Of <(TDelegate>)>)。LambdaExpression 的 NodeType 属性的值为 Lambda。使用 Lambda 工厂方法可创建 LambdaExpression 对象。

LambdaExpression.Body 属性 获取 lambda 表达式的主体。

BindingFlags 枚举 指定控制绑定和由反射执行的成员和类型搜索方法的标志。

0 人点赞