【进阶】C# 委托(Delegate)知识点总结归纳

1. 委托的基本概念

  • 定义:委托是一种类型安全的函数指针,用于封装方法(静态方法或实例方法)。

  • 核心作用:允许将方法作为参数传递,实现回调机制和事件处理。

  • 类型安全:委托在编译时会检查方法签名(参数类型和返回值类型)。

2. 委托的声明与使用

  • 声明语法:
cs 复制代码
delegate [返回类型] DelegateName([参数列表]);

示例:

cs 复制代码
delegate void MyDelegate(string message); // 无返回值,接受一个字符串参数
  • 实例化与调用
cs 复制代码
// 实例化委托(绑定方法)
MyDelegate del = new MyDelegate(MyMethod);

// 调用委托
del("Hello");

static void MyMethod(string msg) {
    Console.WriteLine(msg);
}

3. 多播委托(Multicast Delegate)

  • 特点 :一个委托实例可以绑定多个方法,通过 +=-= 运算符管理方法链。

  • 调用顺序:按添加顺序依次执行所有方法。

  • 返回值:若委托有返回值,只有最后一个方法的返回值会被保留。

cs 复制代码
MyDelegate del1 = Method1;
MyDelegate del2 = Method2;
MyDelegate combined = del1 + del2; // 合并委托
combined("Multi-cast"); // 依次调用 Method1 和 Method2

combined -= del1; // 移除委托

4. 匿名方法与Lambda表达式

  • 匿名方法 (C# 2.0):无需显式定义方法,直接通过 delegate 关键字绑定。
cs 复制代码
MyDelegate del = delegate(string msg) {
    Console.WriteLine("Anonymous: " + msg);
};
  • Lambda表达式(C# 3.0+):简化匿名方法的语法。
cs 复制代码
MyDelegate del = (msg) => Console.WriteLine("Lambda: " + msg);

5. 内置泛型委托

  • Action:无返回值的委托,最多支持16个参数。
cs 复制代码
Action<string> action = (s) => Console.WriteLine(s);
  • Func:有返回值的委托,最后一个泛型参数为返回值类型,最多支持16个参数(不包括返回值类型)。
cs 复制代码
Func<int, int, int> add = (a, b) => a + b;
int result = add(3, 5); // 返回8
  • Predicate :返回 bool 的委托,常用于条件判断,仅支持一个参数。
cs 复制代码
Predicate<int> isEven = num => num % 2 == 0;
bool result = isEven(4); // 返回true

6. 委托与事件

  • 事件本质:事件是基于委托的封装,提供更安全的订阅/取消订阅机制。

  • 事件声明

cs 复制代码
public event EventHandler MyEvent; // EventHandler 是内置委托
  • 触发事件
cs 复制代码
MyEvent?.Invoke(this, EventArgs.Empty); // 安全调用
  • 事件与委托的区别: 事件不能在类外部赋值和调用,只能在声明它的类内部触发(Invoke),外部只能订阅(+=)或取消订阅(-=)。

7. 协变与逆变(Covariance & Contravariance)

  • 协变(Covariance):允许委托返回类型是派生类(更具体类型)。(父类泛型委托装子类泛型委托)
cs 复制代码
delegate Animal GetAnimalDelegate();
GetAnimalDelegate del = GetDog; // Dog 是 Animal 的子类
Animal animal = del();

static Dog GetDog() => new Dog();
  • 逆变(Contravariance):允许委托参数是基类(更通用类型)。(子类泛型委托装父类泛型委托)
cs 复制代码
delegate void ProcessDogDelegate(Dog dog);
ProcessDogDelegate del = ProcessAnimal; // 参数类型为Animal(基类)
del(new Dog());

static void ProcessAnimal(Animal animal) { }

8. 注意事项

  • 内存泄漏 :长期持有委托可能导致对象无法释放,需及时取消订阅(-=)。

  • 线程安全:多线程中委托调用需确保线程同步。

  • 默认值处理:调用空委托会引发异常,需判空:

cs 复制代码
del?.Invoke("Hello");
相关推荐
小小de风呀4 分钟前
de风——【从零开始学C++】(二):类和对象入门(一)
开发语言·c++
浅念-4 分钟前
LeetCode 模拟算法:用「还原过程」搞定编程题的入门钥匙
开发语言·c++·学习·算法·leetcode·职场和发展·模拟
SunnyDays10115 分钟前
C# 如何快速比较 Word 文档并显示差异
c#·对比 word 文档·比较 word 文档
LF男男5 分钟前
TouchPad(单例)
unity·c#
澈2076 分钟前
C++面向对象编程:从封装到实战
开发语言·c++
巨量HTTP9 分钟前
Python 获取动态 iframe 内容(完整解决方案)
开发语言·python
minji...13 分钟前
Linux 网络套接字编程(三)UDP服务器与客户端实现:Windows与Linux通信,新增字典翻译功能的 UDP 通信
linux·服务器·开发语言·网络·windows·算法·udp
人道领域17 分钟前
【Redis实战篇】秒杀系统:一人一单高并发实战(synchronized锁实战与事务失效问题)
java·开发语言·数据库·redis·spring
我不是懒洋洋18 分钟前
PyTorch强化学习实战:从零实现DQN玩转CartPole
笔记
三品吉他手会点灯21 分钟前
C语言学习笔记 - 13.C语言简介 - 回顾本讲内容
c语言·笔记·学习