C#怎么使用委托和事件 C# delegate与event编程方法
技术百科
月夜之吻
发布时间:2026-01-02
浏览: 次 委托是类型安全的函数指针,用于封装方法签名;事件是基于委托的特殊成员,实现发布-订阅模式以解耦通信。
在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 响应、业务解耦、插件扩展都变得更自然。不复杂但容易忽略细节。
# 是一个
# 就像
# 都是
# 尤其是
# 多个
# 则是
# 推荐使用
# 自定义
# 微软
# 数据结构
# public
# ui
# 对象
# String
# if
# int
# void
# class
# 泛型
# 编码
# 指针
# c#
# 委托
# console
# gate
# 线程
# 异步
# 事件
# Event
# var
# this
# NULL
# 多线程
# 封装
# 继承
# try
# catch
# 引用类型
# Delegate
# 运算符
# count
# Lambda
# 匿名方法
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- 如何使用Golang读取日志文件_Golang b
- Win11如何设置开机问候语 Win11修改登录界
- php8.4新语法match怎么用_php8.4m
- 如何在 Go 结构体中正确初始化 map 字段
- c++中如何使用auto关键字_c++11类型推导
- C#如何序列化对象为XML XmlSerializ
- Python随机数生成_random模块说明【指导
- 如何在 Go 项目开发中正确处理本地包导入与远程模
- Python 中将 ISO 8601 时间戳转换为
- PHP cURL GET请求:正确设置请求头与身份
- Go 中 defer 在 goroutine 内部
- Win11怎么开启窗口对齐助手_Windows11
- Windows11如何设置专注助手_Windows
- Win11怎么设置触控板手势_Windows11三
- 如何在 Go 中调用动态链接库(.so)中的函数
- Win11开机速度慢怎么优化_Win11系统启动加
- Windows10怎么备份注册表_Windows1
- Windows任务计划服务异常原因_任务调度失败的
- Win11怎么设置快速访问主页_Windows11
- Golang如何实现基本的用户注册_Golang用
- 如何在Golang中实现服务熔断与限流_Golan
- Win11怎么关闭任务栏小图标_Windows11
- mac怎么退出id_MAC退出iCloud账号与A
- 为什么Go建议使用error接口作为错误返回_Go
- Win11输入法选字框不见了怎么办_Win11输入
- 如何在Golang中指定模块版本_使用go.mod
- 短链接怎么用php递归还原_多层加密链接的处理法【
- 如何使用Golang实现文件追加操作_向已有文件追
- c# 如何深拷贝和浅拷贝
- Mac如何整理桌面文件_Mac使用堆栈功能一键整理
- Win11怎么关闭VBS安全性_Windows11
- 如何在 VS Code 中正确配置并使用 NumP
- c++ std::future和std::prom
- Win11怎么设置默认PDF阅读器 Win11修改
- Win11如何关闭游戏模式 Win11禁用Xbox
- Win11无法拖拽文件到任务栏怎么办_Win11开
- c++23 std::expected怎么用 c+
- php查询数据怎么分组_groupby分组查询配合
- PHP 中如何在函数内持久化修改引用变量的指向
- Go语言中slice追加操作的底层共享机制详解
- Bpmn 2.0的XML文件怎么画流程图
- c++如何连接Redis c++ hiredis库
- 零基础学会Python自动化办公_高效处理Exce
- Windows如何拦截腾讯视频广告_Windows
- 如何更改Windows资源管理器的默认启动位置?(
- 如何从 Go 的 map[string]inter
- Win11怎么连接投影仪_Win11多显示器投屏设
- php嵌入式日志记录怎么实现_php将硬件数据写入
- c++怎么用jemalloc c++替换默认内存分
- C#如何使用XPathNavigator高效查询X

创建委托实例时,可指向静态方法、实例方法,甚至 lambda 表达式:
QQ客服