关于静态事件 static event 的使用

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

public event EventHandler OneEvent;

但是如果这样定义的话就需要进行null判断了
于是很多人这样定义事件:

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

这样就不需要判断null了。
上面的事件你可以把它认为是实例事件. 我们这里要谈的是静态事件(static event).
静态事件和实例事件的区别是加了个static 关键字,可别小看这个关键字,区别大了。
首先静态事件的定义为:

public static event EventHandler OneStaticEvent;

如果想要注册事件的话,那么必须使用类。
在这里插入图片描述
在.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());
        }
    }
}

结果如下:

微软的开发人员只为这些类定义了静态事件,最后一个是我自己写的,我们看下这些究竟是些什么类。
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博客园博客,原文链接:http://www.cnblogs.com/LoveJenny/archive/2013/01/30/2882383.html,如需转载请自行联系原作者

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值