C#怎么使用委托和事件 C# delegate与event编程方法
发布时间:2026-01-02 00:00
发布者:月夜之吻
浏览次数:委托是类型安全的函数指针,用于封装方法签名;事件是基于委托的特殊成员,实现发布-订阅模式以解耦通信。
在C#中,委托(delegate)是类型安全的函数指针,用于封装方法签名;事件(event)则是基于委托的特殊成员,用于实现发布-订阅模式,常用于解耦对象间的通信。掌握它们的关键不是死记语法,而是理解“谁调用谁”“谁响应谁”。
定义和使用自定义委托
委托本质是一
个类,声明时指定返回类型和参数列表。定义后可实例化、赋值、调用:
- 用 delegate 关键字声明,例如:
public delegate void NotifyHandler(string message); - 创建委托实例时,可指向静态方法、实例方法,甚至 lambda 表达式:
NotifyHandler handler = Console.WriteLine;或handler += (m) => Console.WriteLine("收到:" + m); - 调用委托就像调用方法:
handler("操作完成");,若为多播委托(+= 添加多个),会按顺序执行所有绑定方法
用 event 封装委托,实现安全发布
event 是对 delegate 的封装,限制外部代码只能“订阅(+=)”或“取消订阅(-=)”,不能直接调用或赋值,避免误操作破坏内部逻辑:
- 声明 event 必须基于已定义的委托类型:
public event NotifyHandler OnCompleted; - 在类内部触发事件时,需判空再调用:
OnCompleted?.Invoke("任务结束");(推荐用 null 条件运算符) - 外部只能这样响应:
obj.OnCompleted += msg => Console.WriteLine(msg);,不能写obj.OnCompleted = ...或obj.OnCompleted(...)
标准模式:用 EventHandler 提升规范性
微软推荐使用泛型 EventHandler 和继承自 EventArgs 的自定义参数类,让事件更清晰、可扩展:
- 定义事件参数:
public class DataProcessedEventArgs : EventArgs { public int Count { get; } } - 声明事件:
public event EventHandlerDataProcessed; - 触发时传入 sender 和参数:
DataProcessed?.Invoke(this, new DataProcessedEventArgs { Count = 100 }); - 订阅者能明确知道 sender 类型和事件携带的数据结构,利于维护和测试
常见误区与注意事项
实际编码中容易踩坑,注意这几点:
- 事件在多线程环境下可能为空(被其他线程取消订阅),务必用
?.Invoke()或先缓存再判空:var handler = OnCompleted; if (handler != null) handler("ok"); - 记得在不再需要时及时取消订阅(尤其是用匿名方法或 lambda 订阅时),否则可能引发内存泄漏
- 不要在事件触发逻辑里做耗时操作,如需异步处理,应在订阅方自行调度,而非在发布方阻塞
- 委托和 event 都是引用类型,多播委托中任一方法抛异常会中断后续调用,必要时需在内部 try-catch
基本上就这些。委托是机制,事件是约定——用对了,能让 UI 响应、业务解耦、插件扩展都变得更自然。不复杂但容易忽略细节。
# this
# 委托
# Event
# 泛型
# 匿名方法
# 线程
# 多线程
# var
# console
# 对象
# 事件
# Delegate
# 异步
# ui
# 自定义
# 都是
# 就像
# 尤其是
# 多个
# 则是
# 推荐使用
# 能让
# catch
# 微软
# c#
# gate
# String
# NULL
# 运算符
# if
# count
# 封装
# try
# 编码
# int
# void
# Lambda
# 指针
# 数据结构
# 继承
# class
# 引用类型
# public
相关文章:
PHP接收不到OPTIONS预检请求怎么办_处理跨域预检问题解答【说明】
Win11怎么关闭专注助手 Win11关闭免打扰模式设置【操作】
composer怎么配置多渠道下载源_composer多repositories仓库优先级设置【技巧】
如何解决 Jenkins 中 pytest 参数化测试被跳过的问题
PHP怎么接收前端传的文件路径_处理文件路径参数接收方法【汇总】
悟空浏览器小说入口_全网热门网文免费阅读
Python进程间通信机制_queue解析【教程】
Java 中实现智能金额单位(k/M/B)的简洁封装方案
标题:如何同时实现 Scroll-Snap 与平滑背景色过渡效果
移动端中的巅峰营销--微信营销
Sugar Conflict X在哪下载-Sugar Conflict X下载地址分享
Win11怎么设置默认输入法 Win11固定中文输入法【步骤】
iPhone 17 Pro系列充电有异响?官方回应来了
通义万相电商主图怎么用_通义万相电商主图使用方法详细指南【教程】
海外怎么下载国内版抖音 国际版抖音TikTok国内使用教程【详解】
如何使用Golang实现字符串操作_Golang字符串拼接与切割方法
高德导航视角跟随是什么意思_视角跟随模式功能详解
如何正确解析带UTC偏移量的日期字符串并转换为标准ISO UTC格式
如何使用Golang写入文件数据_通过os和bufio写入文本和二进制
Chrome浏览器标签页分组怎么用_谷歌浏览器整理标签页技巧【效率】
如何从SQL数据库动态填充下拉菜单(Dropdown)
Owlcat官宣推出自家启动器 玩家社区炸锅:不需要!
相机工业百年产量不敌一家手机厂!OPPO已累计交付55亿颗摄像头
解决 Telegram Web View 在 iOS 中软键盘遮挡输入框的问题
如何使用PHP根据用户会话导出专属SQL数据到Excel文件
HTML 中如何正确使用模板变量为元素的 name 属性赋值
php订单日志怎么记录积分变动_php记录订单积分变化日志说明【说明】
composer如何配置本地path仓库进行包开发_composer本地路径依赖映射【技巧】
哈迪斯入门双流派全解 通关加点技巧分享
sublime怎么配置elixir开发环境_sublime安装elixir-ls插件高亮设置【方案】
相关栏目:
【
行业资讯17850 】
【
软件资源51899 】
【
网站技术89748 】
【
百度推广44206 】
【
网络营销84187 】
【
运营推广93002 】
【
AI优化91086 】
【
网络优化117696 】
【
网址导航107142 】






