当前位置: 首页 > news >正文

.NET/C# 使用反射注册事件

使用反射,我们可以很容易地在运行时调用一些编译时无法确定的属性、方法等。那么如何注册事件呢?
本文将介绍如何使用反射注册事件。


本文内容

      • 不使用反射
      • 使用反射
      • 安全地使用反射
        • 参考资料

不使用反射

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

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

相关文章:

  • .NET/C# 判断某个类是否是泛型类型或泛型接口的子类型
  • .NET/C# 使用反射调用含 ref 或 out 参数的方法
  • WPF 多线程 UI:设计一个异步加载 UI 的容器
  • .NET 命令行参数包含应用程序路径吗?
  • 分析现有 WPF / Windows Forms 程序能否顺利迁移到 .NET Core 3.0(使用 .NET Core 3.0 Desktop API Analyzer )
  • C# 空合并操作符(??)不可重载?其实有黑科技可以间接重载!
  • UWP 轻量级样式定义(Lightweight Styling)
  • 预编译框架,开发高性能应用 - 课程 - 微软技术暨生态大会 2018
  • 将 UWP 中 CommandBar 的展开方向改为向下展开
  • .NET 中创建支持集合初始化器的类型
  • .NET 中让 Task 支持带超时的异步等待
  • .NET/C# 中你可以在代码中写多个 Main 函数,然后按需要随时切换
  • WPF 的 ElementName 在 ContextMenu 中无法绑定成功?试试使用 x:Reference!
  • WPF 中的 NameScope
  • Windows 下的高 DPI 应用开发(UWP / WPF / Windows Forms / Win32)
  • 深入了解以太坊
  • [case10]使用RSQL实现端到端的动态查询
  • - C#编程大幅提高OUTLOOK的邮件搜索能力!
  • Docker容器管理
  • If…else
  • Invalidate和postInvalidate的区别
  • JavaScript 无符号位移运算符 三个大于号 的使用方法
  • Java精华积累:初学者都应该搞懂的问题
  • jquery cookie
  • LeetCode18.四数之和 JavaScript
  • MD5加密原理解析及OC版原理实现
  • mysql 5.6 原生Online DDL解析
  • MySQL数据库运维之数据恢复
  • React的组件模式
  • TCP拥塞控制
  • ubuntu 下nginx安装 并支持https协议
  • Vue 重置组件到初始状态
  • Vue2 SSR 的优化之旅
  • vue从创建到完整的饿了么(11)组件的使用(svg图标及watch的简单使用)
  • 算法-图和图算法
  • 温故知新之javascript面向对象
  • 再次简单明了总结flex布局,一看就懂...
  • 阿里云服务器如何修改远程端口?
  • 国内开源镜像站点
  • ​Java并发新构件之Exchanger
  • #Linux(权限管理)
  • #我与Java虚拟机的故事#连载02:“小蓝”陪伴的日日夜夜
  • #我与Java虚拟机的故事#连载17:我的Java技术水平有了一个本质的提升
  • (cljs/run-at (JSVM. :browser) 搭建刚好可用的开发环境!)
  • (超详细)语音信号处理之特征提取
  • (非本人原创)我们工作到底是为了什么?​——HP大中华区总裁孙振耀退休感言(r4笔记第60天)...
  • (附源码)ssm旅游企业财务管理系统 毕业设计 102100
  • (四)【Jmeter】 JMeter的界面布局与组件概述
  • (一)VirtualBox安装增强功能
  • (转)ABI是什么
  • (转载)虚函数剖析
  • .net core 客户端缓存、服务器端响应缓存、服务器内存缓存
  • .NET core 自定义过滤器 Filter 实现webapi RestFul 统一接口数据返回格式
  • .net MVC中使用angularJs刷新页面数据列表
  • .NET6 开发一个检查某些状态持续多长时间的类