掌握 C# 中的委托与事件机制

C# 中的委托和事件为开发者提供了处理回调、异步编程以及发布订阅模式的强大工具。委托与事件机制在实际应用中非常常见,特别是在事件驱动编程和 GUI 应用中。本文将带你深入理解委托的定义、匿名方法、Lambda 表达式、事件机制以及多播委托的使用。


1. 委托(Delegate)的定义与使用

委托 是一种类型安全的函数指针,可以用于引用一个或多个方法。通过委托,可以将方法作为参数传递给其他方法,从而实现回调和灵活的行为封装。

委托的定义

委托的定义类似于定义方法签名,它指定了可以被引用的方法的返回类型和参数列表。

// 定义一个委托
public delegate void PrintDelegate(string message);

// 使用委托
public class Printer
{
    public void PrintMessage(PrintDelegate printDelegate, string message)
    {
        printDelegate(message);  // 调用委托
    }
}

public class Program
{
    public static void PrintToConsole(string message)
    {
        Console.WriteLine(message);
    }

    public static void Main()
    {
        Printer printer = new Printer();
        PrintDelegate printDelegate = PrintToConsole;  // 将方法赋值给委托
        printer.PrintMessage(printDelegate, "Hello, Delegates!");  // 输出:Hello, Delegates!
    }
}

在上面的示例中,PrintDelegate 是一个委托类型,它可以引用任何具有 void 返回类型且接受 string 参数的方法。我们将 PrintToConsole 方法赋值给委托实例,并通过委托调用该方法。


2. 匿名方法与 Lambda 表达式

C# 提供了匿名方法和 Lambda 表达式来简化委托的使用,避免显式定义命名方法。

匿名方法

匿名方法允许你直接将方法逻辑嵌入到委托实例化过程中,而无需创建一个命名方法。

PrintDelegate printDelegate = delegate (string message)
{
    Console.WriteLine(message);
};

printDelegate("Hello, Anonymous Methods!");  // 输出:Hello, Anonymous Methods!

Lambda 表达式

Lambda 表达式是匿名方法的简写形式,语法更加简洁。它使用 => 运算符来分隔参数和方法体。

PrintDelegate printDelegate = (message) => Console.WriteLine(message);

printDelegate("Hello, Lambda Expressions!");  // 输出:Hello, Lambda Expressions!

Lambda 表达式在委托、事件和 LINQ 查询中广泛应用,能够极大简化代码编写。


3. 事件机制(Event)

事件 是基于委托的一种特殊机制,通常用于实现发布/订阅模式。事件是对象之间通信的一种方式,允许对象响应特定的状态变化或动作。

事件的定义

事件本质上是对委托的封装,防止订阅者直接调用委托,只允许通过 += 和 -= 来订阅或取消订阅事件。

public class Button
{
    // 定义一个事件
    public event EventHandler Click;

    public void OnClick()
    {
        if (Click != null)
        {
            Click(this, EventArgs.Empty);  // 触发事件
        }
    }
}

public class Program
{
    public static void ButtonClicked(object sender, EventArgs e)
    {
        Console.WriteLine("Button clicked!");
    }

    public static void Main()
    {
        Button button = new Button();
        button.Click += ButtonClicked;  // 订阅事件
        button.OnClick();  // 输出:Button clicked!
    }
}

在这个示例中,Click 是一个事件,使用 EventHandler 委托。当 OnClick 方法被调用时,事件被触发,所有订阅该事件的方法都会被执行。


4. 多播委托

多播委托 是指一个委托可以同时引用多个方法。每当该委托被调用时,所有被引用的方法都会依次执行。多播委托在事件处理中非常有用,因为事件通常会有多个订阅者。

public delegate void NotifyDelegate(string message);

public class Program
{
    public static void PrintToConsole(string message)
    {
        Console.WriteLine($"Console: {message}");
    }

    public static void PrintToFile(string message)
    {
        Console.WriteLine($"File: {message} (simulated)");
    }

    public static void Main()
    {
        NotifyDelegate notifyDelegate = PrintToConsole;
        notifyDelegate += PrintToFile;  // 添加另一个方法

        notifyDelegate("Multicast Delegate Example");
        // 输出:
        // Console: Multicast Delegate Example
        // File: Multicast Delegate Example (simulated)
    }
}

在此示例中,notifyDelegate 委托同时引用了两个方法。当 notifyDelegate 被调用时,两个方法都会依次执行。这就是多播委托的功能。

  • 注意:如果多播委托中包含返回值的方法,只有最后一个方法的返回值会被保留,其余的返回值会被忽略。

结论

委托和事件是 C# 编程中的重要概念,它们使得方法可以作为对象进行传递和处理。在事件驱动编程中,委托和事件的结合非常强大,可以帮助我们构建松耦合、可扩展的程序。

  • 委托 允许将方法作为参数传递,使得代码更加灵活。
  • 匿名方法和 Lambda 表达式 简化了委托的使用,使代码更简洁。
  • 事件机制 为实现发布/订阅模式提供了强大的工具,常用于 GUI 或者异步任务处理。
  • 多播委托 允许一个委托引用多个方法,是事件机制的基础。

通过掌握这些核心概念,你可以编写出更具扩展性和灵活性的 C# 程序。如果你对某个部分有进一步的疑问或需要深入讨论,欢迎继续交流!


这篇博客为你介绍了 C# 中委托与事件的基本概念和应用。如果你有任何问题或者需要更多细节,欢迎留言或者联系我!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/888081.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

015 品牌关联分类

文章目录 后端CategoryBrandEntity.javaCategoryBrandController.javaCategoryBrandServiceImpl.javaCategoryServiceImpl.javaBrandServiceImpl.java删除 npm install pubsub-jsnpm install --save pubsub-js这个错误是由于在尝试安装 pubsub-js 时,npm 发现了项目…

数据结构(栈和队列的实现)

1. 栈(Stack) 1.1 栈的概念与结构 栈是一种特殊的线性表,其只允许固定的一段插入和删除操作;进行数据插入和删除的一段叫做栈顶,另一端叫栈底;栈中的元素符合后进先出LIFO(Last In First Out&…

C++——模拟实现vector

1.查看vector的源代码 2.模拟实现迭代器 #pragma oncenamespace jxy {//模板尽量不要分离编译template <class T>class vector{public:typedef T* iterator;//typedef会受到访问限定符的限制typedef const T* const_iterator;//const迭代器是指向的对象不能修改&#xf…

透明物体的投射和接收阴影

1、让透明度测试Shader投射阴影 &#xff08;1&#xff09;同样我们使用FallBack的形式投射阴影&#xff0c;但是需要注意的是&#xff0c;FallBack的内容为&#xff1a;Transparent / Cutout / VertexLit&#xff0c;该默认Shader中会把裁剪后的物体深度信息写入到 阴影映射纹…

毕业设计_基于springboot+ssm+bootstrap的旅游管理系统【源码+SQL+教程+可运行】【41001】.zip

毕业设计_基于springbootssmbootstrap的旅游管理系统【源码SQL教程可运行】【41001】.zip 下载地址&#xff1a; https://download.csdn.net/download/qq_24428851/89828190 管理系统 url: http://localhost:8080/managerLoginPageuser: admin password: 123 用户门户网站…

【设计模式-解释模式】

定义 解释器模式是一种行为设计模式&#xff0c;用于定义一种语言的文法&#xff0c;并提供一个解释器来处理该语言的句子。它通过为每个语法规则定义一个类&#xff0c;使得可以将复杂的表达式逐步解析和求值。这种模式适用于需要解析和执行语法规则的场景。 UML图 组成角色…

SPDK从安装到运行hello_world示例程序

SPDK从安装到运行示例程序 #mermaid-svg-dwdwvhrJiTcgTkVf {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-dwdwvhrJiTcgTkVf .error-icon{fill:#552222;}#mermaid-svg-dwdwvhrJiTcgTkVf .error-text{fill:#552222;s…

android compose ScrollableTabRow indicator 指示器设置宽度

.requiredWidth(30.dp) Box(modifier Modifier.background(Color.LightGray).fillMaxWidth()) {ScrollableTabRow(selectedTabIndex selectedTabIndex, // 默认选中第一个标签containerColor ColorPageBg,edgePadding 1.dp, // 内容与边缘的距离indicator { tabPositions…

【本地缓存】Java 中的 4 种本地缓存

目录 1、手写一个简单的本地缓存1.1、封装缓存实体类1.2、创建缓存工具类1.3、测试 2、Guava Cache2.1、Guava Cache 简介2.2、入门案例2.2.1、引入 POM 依赖2.2.2、创建 LoadingCache 缓存 2.3、Guava Cache 的优劣势和适用场景 3、Caffeine3.1、Caffeine 简介3.2、对比 Guava…

图的基本概念 - 离散数学系列(五)

目录 1. 图的定义 节点与边 2. 度与路径 节点的度 路径与圈 3. 图的连通性 连通图与非连通图 强连通与弱连通 连通分量 4. 实际应用场景 1. 社交网络 2. 城市交通系统 3. 网络结构 5. 例题与练习 例题1&#xff1a;节点的度 例题2&#xff1a;判断连通性 练习题…

linux基础 超级笔记

1.Linux系统的组成 Linux系统内核&#xff1a;提供系统最核心的功能&#xff0c;如软硬件和资源调度。 系统及应用程序&#xff1a;文件、任务管理器。 2.Linux发行版 通过修改内核代码自行集成系统程序&#xff0c;即封装。比如Ubuntu和centos这种。不过基础命令是完全相…

【瑞昱RTL8763E】刷屏

1 显示界面填充 用户创建的各个界面在 rtk_gui group 中。各界面中 icon[]表对界面进行描述&#xff0c;表中的每个元素代表一 个显示元素&#xff0c;可以是背景、小图标、字符等&#xff0c;UI_WidgetTypeDef 结构体含义如下&#xff1a; typedef struct _UI_WidgetTypeDef …

vite学习教程03、vite+vue2打包配置

文章目录 前言一、修改vite.config.js二、配置文件资源/路径提示三、测试打包参考文章资料获取 前言 博主介绍&#xff1a;✌目前全网粉丝3W&#xff0c;csdn博客专家、Java领域优质创作者&#xff0c;博客之星、阿里云平台优质作者、专注于Java后端技术领域。 涵盖技术内容&…

【深度强化学习基础】(一)基本概念

【深度强化学习基础】&#xff08;一&#xff09;基本概念 一、概率论基础知识二、强化学习领域术语三、强化学习中两个随机性的来源&#xff1a;四、rewards以及returns五、Value Functions1.Action-Value Function Q π ( s , a ) Q_\pi(s,a) Qπ​(s,a)2.State-Value Funct…

【高等数学学习记录】函数的极限

一、知识点 &#xff08;一&#xff09;知识结构 #mermaid-svg-Dz0Ns0FflWSBWY50 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-Dz0Ns0FflWSBWY50 .error-icon{fill:#552222;}#mermaid-svg-Dz0Ns0FflWSBWY50 .erro…

影刀---如何进行自动化操作

本文不是广告&#xff0c;没有人给我宣传费&#xff0c;只是单纯的觉得这个软件很好用 感谢大家的多多支持哦 本文 1.基本概念与操作&#xff08;非标准下拉框和上传下载&#xff09;非标准对话框的操作上传对话框、下载的对话框、提示的对话框 2.综合案例3.找不到元素怎么办&a…

Leecode刷题之路第12天之整数转罗马数字

题目出处 12-整数转罗马数字-题目出处 题目描述 个人解法 思路&#xff1a; todo 代码示例&#xff1a;&#xff08;Java&#xff09; todo复杂度分析 todo 官方解法 12-整数转罗马数字-官方解法 方法1&#xff1a;模拟 思路&#xff1a; 代码示例&#xff1a;&#xff08…

class 032 位图

这篇文章是看了“左程云”老师在b站上的讲解之后写的, 自己感觉已经能理解了, 所以就将整个过程写下来了。 这个是“左程云”老师个人空间的b站的链接, 数据结构与算法讲的很好很好, 希望大家可以多多支持左程云老师, 真心推荐. 左程云的个人空间-左程云个人主页-哔哩哔哩视频…

SpringBoot项目:前后端打包与部署(使用 Maven)

文章目录 IDEA后端打包与部署&#xff08;使用 Maven&#xff09;1. 确保 Maven 已安装&#xff0c;并引入 pom 插件2. 清理并安装项目3. 定位生成的 JAR 包和配置文件4. 创建部署文件夹5. 上传到服务器 前端打包与部署&#xff08;使用 npm&#xff09;1. 确保 Node.js 和 npm…

Oracle 数据库安装和配置详解

Oracle 数据库安装和配置详解 Oracle 数据库是一款功能强大、广泛使用的企业级关系数据库管理系统 (RDBMS)&#xff0c;适用于处理大型数据库和复杂事务。本文将介绍如何在 Linux 和 Windows 环境下安装 Oracle 数据库&#xff0c;并对其进行基本配置&#xff0c;帮助开发者快…