.NET/C#如何使用反射注册事件详解

时间:2021-05-28

前言

通过放射,可以在运行时获得.NET中每一个类型(包括类、结构、委托、接口和枚举等)的成员,包括方法、属性、事件,以及构造函数等。还可以获得每个成员的名称、限定符和参数等。有了反射,即可对每一个类型了如指掌。如果获得了构造函数的信息,即可直接创建对象,即使这个对象的类型在编译时还不知道。那么如何注册事件呢?

本文将介绍如何使用反射注册事件。下面话不多说了,来一起看看看详细的介绍吧

不使用反射

例如,我们希望反射的类型是这样的:

public class Walterlv{ public event EventHandler BlogPublished;}

那么只需要使用如下代码即可完成事件的注册:

var walterlv = new Walterlv();walterlv += Walterlv_BlogPublished;public void Walterlv_BlogPublished(object sender, EventHandler handler){}

使用反射

而如果使用反射,则是:

var walterlv = new Walterlv();var eventInfo = typeof(Walterlv).GetEvent(nameof(BlogPublished));var handler = new EventHandler(Walterlv_BlogPublished);eventInfo.AddEventHandler(walterlv, handler);

当然,实际使用的时候,如果能访问到 Walterlv 类型,当然也不会去用到反射,所以通常情况是这样的:

public void AddHandler<T>(T instance, string eventName, EventHandler handler){ var eventInfo = instance.GetType().GetEvent(eventName); eventInfo.AddEventHandler(instance, handler);}

安全地使用反射

虽然以上方式使用了反射成功注册了事件,但实际上我们的参数中传入了一个特定类型的委托 EventHandler。实际上事件的委托种类非常多。

在委托中,即便签名完全相同,也不是同一个委托类型。如果传入的参数类型改为 EventHandler<EventArgs> ,或者 BlogPublished 事件的类型改为 EventHandler<EventHandler>,虽然实际上这两个委托的签名是兼容的,但其委托类型不同,依然是不能互相转换的。你会在运行时遇到一下异常:


▲ 委托无法转换

所以我们必须有一些更安全的方式来注册事件。

正常情况下,我们转换一个签名兼容的委托是使用构造函数:

public EventHandler ConvertDelegate(EventHandler<EventArgs> handler){ return new EventHandler(handler);}

那么在反射中,我们需要使用 Delegate.CreateDelegate 创建指定类型的委托。

public void AddHandler<T>(T instance, string eventName){ var eventInfo = instance.GetType().GetEvent(eventName); var methodInfo = GetType().GetMethod(nameof(Walterlv_BlogPublished)); var @delegate = Delegate.CreateDelegate(eventInfo.EventHandlerType, this, methodInfo); eventInfo.AddEventHandler(instance, @delegate);}public void Walterlv_BlogPublished(object sender, EventHandler handler){}

这里,Delegate.CreateDelegate 的作用就是执行委托类型的转换。我在 .NET Core/Framework 创建委托以大幅度提高反射调用的性能 中也提到过这个方法。

参考资料

c# - AddEventHandler using reflection - Stack Overflow

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。

声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。

相关文章