C# 中的回调函数

在 C# 中,"回调函数"(Callback Function)这一概念主要是通过**委托(Delegate)**来实现的。委托是 C# 的核心特性之一,它使得代码具有极大的灵活性和可扩展性。


C# 中的回调函数(Callback Function)

1. 什么是回调函数?

回调函数本质上就是一个作为参数传递给另一个函数的函数

  • 调用方(Caller):是接收这个函数作为参数,并在内部某个时刻执行(调用)它的那个函数或方法。
  • 回调函数(Callback):就是那个被传递进去并等待被执行的函数。

核心思想: 允许一个较低级别的函数(或类)在完成某个操作时,通知或执行由较高级别函数(或类)提供的特定操作。这实现了控制的反转

2. C# 如何实现回调?------ 委托(Delegate)

在 C# 中,**委托(Delegate)**是实现回调机制的"基石"或"函数指针"。

a. 委托的定义

委托是一种类型安全 的函数指针,它定义了一个方法的签名(包括返回类型和参数列表)。

示例代码:定义一个委托

csharp 复制代码
// 1. 定义委托
// 想象它是一个"合同",规定了回调函数的签名:
// 必须接收一个 string 参数,并返回 void。
public delegate void PrintMessage(string message);
b. 回调函数的实现(订阅者)

回调函数是遵循委托签名要求的任何方法。

示例代码:实现回调函数

csharp 复制代码
public class Subscriber
{
    // 这是一个符合 PrintMessage 委托签名的方法
    public void DisplayInfo(string info)
    {
        Console.WriteLine($"[信息]:{info}");
    }

    // 另一个符合 PrintMessage 委托签名的方法
    public static void LogError(string error)
    {
        Console.WriteLine($"[错误日志]:{error} (来自静态方法)");
    }
}
c. 调用方的设计(发布者)

调用方(执行回调的函数)需要接收一个委托实例作为参数。

示例代码:调用方

csharp 复制代码
public class Caller
{
    // 核心方法:它接收一个委托(即回调函数的引用)作为参数
    public void ProcessTask(string data, PrintMessage callbackFunc)
    {
        Console.WriteLine("---- 开始处理任务 ----");

        if (string.IsNullOrEmpty(data))
        {
            // 在特定条件(数据为空)下,调用方执行回调函数
            // 注意:这里调用的是 LogError
            callbackFunc.Invoke("数据输入为空,无法继续!");
        }
        else
        {
            // 在特定条件(处理成功)下,调用方执行回调函数
            // 注意:这里调用的是 DisplayInfo
            callbackFunc.Invoke($"成功处理数据:{data.ToUpper()}");
        }
        
        Console.WriteLine("---- 任务处理结束 ----");
    }
}
d. 使用和调用

在主程序中,创建委托实例,并将其传递给调用方。

示例代码:主程序调用

csharp 复制代码
public class Program
{
    public static void Main()
    {
        Subscriber sub = new Subscriber();
        Caller caller = new Caller();

        // **步骤 1: 将回调函数(方法)绑定到委托实例**
        // 创建委托实例,指向 Subscriber 实例的 DisplayInfo 方法
        PrintMessage displayCallback = sub.DisplayInfo;

        // **步骤 2: 将委托(回调的引用)传递给调用方**
        Console.WriteLine("--- 第一次调用(正常数据)---");
        caller.ProcessTask("hello world", displayCallback);
        // 输出将由 DisplayInfo 方法格式化

        Console.WriteLine("\n--- 第二次调用(空数据)---");
        // 也可以直接在调用时创建并传递委托实例
        caller.ProcessTask("", sub.DisplayInfo); 

        // 也可以使用不同的回调函数(例如静态方法 LogError)
        Console.WriteLine("\n--- 第三次调用(使用 LogError)---");
        caller.ProcessTask("Another Data", Subscriber.LogError);
    }
}

3. 简化方式:Func、Action 和 Lambda 表达式

在现代 C# 中,为了避免每次都定义新的委托类型,我们通常使用内置的泛型委托:

  • Action:用于没有返回值的回调函数(最多支持 16 个参数)。
  • Func<TResult> :用于有返回值的回调函数(TResult 是返回类型,前面的参数是输入参数)。

同时,Lambda 表达式(匿名函数)使得回调函数的定义更加简洁:

使用 Func 和 Lambda 的示例:

csharp 复制代码
public void CalculateAndReport(int num1, int num2, Func<int, int, int> calculationLogic)
{
    // calculationLogic 是一个回调函数,接收两个 int,返回一个 int
    int result = calculationLogic.Invoke(num1, num2);
    Console.WriteLine($"计算结果是: {result}");
}

public static void Main()
{
    // 传递一个 Lambda 表达式作为回调函数(执行加法)
    CalculateAndReport(10, 5, (a, b) => a + b); // 输出: 15

    // 传递另一个 Lambda 表达式作为回调函数(执行乘法)
    CalculateAndReport(10, 5, (x, y) => x * y); // 输出: 50
}

4. 回调的应用场景

回调机制在 C# 中有着广泛的应用,尤其是在以下场景:

场景 描述 C# 实现方式
事件处理 当特定事件(如按钮点击、文件下载完成)发生时,通知并执行订阅者的方法。 事件(Event),它本质上是委托的特殊封装。
异步操作 在 I/O 操作(如网络请求、文件读写)完成时,执行后续处理逻辑。 Taskawait/async(底层也依赖回调机制)。
自定义逻辑 允许用户向通用算法(如排序、过滤)中注入自定义的比较或筛选逻辑。 List<T>.Sort() 接收一个 Comparison<T> 委托(即回调)。
插件机制 设计框架时,预留接口让外部模块在特定时机执行其功能。 委托或接口。

总结来说,在 C# 中,委托是回调函数机制的官方实现,它提供了一种灵活、类型安全的方式,来将代码的执行权从一个函数(调用方)传递给另一个函数(回调函数)。

相关推荐
爱上妖精的尾巴18 小时前
5-26 WPS JS宏数组元素添加删除应用
开发语言·前端·javascript·wps·js宏
是谁眉眼18 小时前
wpsapi
前端·javascript·html
短视频矩阵源码定制18 小时前
矩阵系统全面解析:构建智能营销体系的核心引擎
java·人工智能·矩阵·aigc·视频
谅望者18 小时前
Flexbox vs Grid:先学哪一个?CSS 布局完全指南(附可视化示例)
前端·css·html·css3·css布局·css flexbox·css grid
蛮三刀酱18 小时前
代码复杂度的代价远比你想象得大
java
老华带你飞19 小时前
商城推荐系统|基于SprinBoot+vue的商城推荐系统(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·毕设·商城推荐系统
JS.Huang19 小时前
【JavaScript】Pointer Events 与移动端交互
前端·javascript
花北城19 小时前
【C#】List快速检查重复数据
开发语言·c#
一 乐19 小时前
物业管理系统|小区物业管理|基于SprinBoot+vue的小区物业管理系统(源码+数据库+文档)
java·前端·数据库·vue.js·spring boot·后端
H_HX12619 小时前
vue3 - 图片放大镜效果实现
前端·vue.js·vue3·vueuse·图片放大镜