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



    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 枚举
    指定控制绑定和由反射执行的成员和类型搜索方法的标志。

  • 相关阅读:
    英语_词汇_同意辨析
    英语_网站_写作工具
    英语词汇_难词易忘
    IDEA配置技巧 | 去除idea方法形参赋值时的变量提示
    更换Android studio中的SDK和AVD位置
    CSS/CSS3 | P4-选择器优先级权重
    二叉树的性质
    线性表练习
    前插法建立链表
    格式化文件和数据块读写函数
  • 原文地址:https://www.cnblogs.com/liulun/p/1573271.html
Copyright © 2020-2023  润新知