• 关于静态事件 static event 的二三事


    很多人都知道事件,一般定义一个事件是这样的写法

    public event EventHandler OneEvent;

    但是如果这样定义的话就需要进行null判断了

    825E6B5E43014CC898E20B3BFDFC1B00

    于是很多人这样定义事件:

    public event EventHandler OneEventNeverNull = (o, e) => { };

    这样就不需要判断null了。

    上面的事件你可以把它认为是实例事件. 我们这里要谈的是静态事件(static event).

    静态事件和实例事件的区别是加了个static 关键字,可别小看这个关键字,区别大了。

    首先静态事件的定义为:

    public static event EventHandler OneStaticEvent;

    如果想要注册事件的话,那么必须使用类。

    image

    在.net 的事件里面,绝大多数是实例事件,但是却存在极少数的静态事件。

    关于静态事件StackOverflow 上面有篇文章讨论它

    地址如下:http://stackoverflow.com/questions/7045595/how-do-static-events-compare-to-non-static-events-in-c

    我个人认为答案都不错,选了几个,试着翻译如下:

    大部分的面向对象程序都可以被认为是消息的传递。

    一个方法调用就是调用者发送给被调用者的携带参数的一个消息,和一个使用return value 做为返回值的消息。

    一个事件是从source到订阅者的一个消息。因此就有两个实例参与其中,一个用来发送消息,另一个接收它。

    但是静态事件,没有发送的实例(只是一个类型,它可能是也可能不是一个class)。

    使用 静态事件的时候,要谨记,当一个对象订阅了一个静态事件,事件就持有了这个对象的引用,这就意味着你必须非常小心的显式的取消订阅事件,因为静态事件保留了对象引用,所以这些对象就永远不会被回收,你很可能碰到内存泄漏。

    这些答案都说明了静态事件的一些特性。总结下:

    1:静态事件没有发送的实例对象。

    2:静态事件必须取消订阅,否则对象无法释放,事实上任何事件都应该取消订阅,有始有终。

    对于第二点没有什么好考虑的,可是第一点?

    静态事件没有发送的实例?

    a:这代表静态事件不需要实例就可以,它代表的是类级别的事件。

    b:这代表静态事件没有实例,或者说无法获取这个类的实例,所以使用静态事件。

    从这两个理解,我们可以知道什么时候应该使用静态事件了。

    a:从面向对象的角度,如果一个对象只想获取事件的通知,而不关心是谁发送的,它只关心事件发生。

    b:这个类没有实例?,什么类没有实例???,静态类,抽象类,或者是一些特殊的类。

    好了,我不卖关子了,让我们看看framework里面的静态事件吧,代码如下:

    Assembly[] assemblies = AppDomain.CurrentDomain.GetAssemblies();
    
    foreach (Assembly assembly in assemblies)
    {
        foreach (Type type in assembly.GetTypes())
        {
            EventInfo[] events = type.GetEvents(BindingFlags.Public | BindingFlags.Static);
            if (events.Length > 0)
            {
                Console.WriteLine(type.ToString());
            }
        }
    }
    结果如下:
    00815F6C3CA84DEE9B7FA34A64EB9BB2
     

    微软的开发人员只为这些类定义了静态事件,最后一个是我自己写的,我们看下这些究竟是些什么类。

    System.Console

    静态类,无法获取类的实例,所以静态事件。


    System.Threading.Tasks.TaskScheduler

    抽象类,它的静态事件定义如下:

    //     当出错的 System.Threading.Tasks.Task 的未观察到的异常将要触发异常升级策略时发生,默认情况下,这将终止进程。
            public static event EventHandler<UnobservedTaskExceptionEventArgs> UnobservedTaskException;

    只要应用程序注册了 UnobservedTaskException 事件,任何TaskScheduler的子类在发生exception的时候,都会通知到应用程序。


    System.Diagnostics.Eventing.EventProviderDataStream

    没找到。。。


    System.Diagnostics.Contracts.Contract

    静态类,无法获取类的实例,所以静态事件。


    System.Windows.Forms.Application

    密封类,构造函数为私有,外界无法直接获取Application 的实例。


    System.Windows.Forms.ToolStripManager

    密封类,构造函数为私有,外界无法直接获取实例。


    System.ComponentModel.TypeDescriptor

    密封类,构造函数为私有,外界无法直接获取实例。


    Microsoft.Win32.SystemEvents

    密封类,构造函数为私有,外界无法直接获取实例。

    System.Net.NetworkInformation.NetworkChange

    密封类,构造函数为私有,外界无法直接获取实例。

    StaticEventDemo.Test

    出现的理由:在某天夜晚,我自己想当然的写下的没有经过思考的一个静态事件。

    你知道微软使用静态事件的策略了吗?,只有在不能获取实例的情况下才选择静态事件。

    有趣的一件事:本人在baidu下搜索:static event, 看到了一篇文章:

    net 在类中的Event事件,为什么可以定义为static?而委托类型却不可以

    也可以在这里http://www.2cto.com/kf/201110/108280.html看到代码。

    我看到了下面的这段:

    private void button1_Click(object sender, EventArgs e)
    {
                clsEvent clsevent = new clsEvent();
                //改变属性,从而激发事件。
                clsevent.StrContent = this.textBox1.Text;

    我很想说,在改变属性前,是不是要注册点什么啊,我想这就是他为什么定义为static event的原因吧。

    总结:

    在无法获取实例的情况下,又想获得通知,应该使用静态事件。

    不管发送者是谁,只考虑接收事件。

    静态事件一定要取消订阅,否则内存泄漏。

    想要获取静态类的通知时,使用静态事件。

    想要得到静态属性和方法的通知的时候。

    作者:LoveJenny
    本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
  • 相关阅读:
    Typora设置插入图片居左
    android模拟器横屏后屏幕不自动旋转的问题
    (转) word: 通过Office 2007发布博客
    Typora导出文件到github指令
    c++函数式编程 笔记
    关于PostMan的一个坑
    VS Code中Golang远程开发的调试和测试
    在GoLand/Pycharm中安装Copilot时遇到“waiting for github authentication”
    安装kubetest报错 "ambiguous import: found package cloud.google.com/go/storage in multiple modules:"
    window 客户端抓包详解
  • 原文地址:https://www.cnblogs.com/LoveJenny/p/2882383.html
Copyright © 2020-2023  润新知