C# 委托详解(Delegate)

2024-08-28 20:44

本文主要是介绍C# 委托详解(Delegate),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

引言

在 C# 编程当中,委托(Delegate)是一种特殊的类型,它允许将方法作为参数传递给其他方法,或者将方法作为返回值返回,这种特性使得委托成为实现回调函数、事件处理等,所有的委托都派生自System.Delegate类。

C#中的委托类似于C 或 C++中函数的指针。委托是存在对某个方法的引用的一种引用类型变量,引用可以在运行的时候被打断,当事件出现,就必然有委托的影子,而有委托的时候却不一定会有事件出现。


委托的概念

委托是一种引用类型变量,存储某一个方法的内存地址

委托的格式

public delegate 返回值类型 MyDelegate();
//这里面的Mydelegate就是一个委托类型

委托的使用步骤

  1. 声明一个委托(函数指针)
  2. 创建委托对象
  3. 创造符合委托格式的函数。(指针指向的函数)
  4. 将函数名称赋值给委托。

这里举个例子吧:

using System;
namespace 练习
{internal class Program{public delegate int Count(int a, int b);public static int Add(int a, int b){return a + b;}static void Main(string[] args){Count c1 = new Count(Add);int c = c1(1, 2);Console.WriteLine(c);Console.Read();}}
}

在这串代码里面 定义了一个名为 "Count"委托类型,这里 Count 可以引用任何接收两个int参数并且返回一个 int 的方法

定义了一个名为 "Add" 的静态方法,它接受了两个 int 参数并且返回它们的和

在Main方法里面,创建了一个Count类型的委托实例 c1 ,并将其初始化引用为 Add 方法,并传入了这两个参数

调用委托 c1,传入两个参数1和2,由于委托引用 Add 方法,所以这行代码实际上调用了Add方法,并传入了这两个参数

小贴士

        使用委托时,我们一般采用两种情况

  1. 直接调用委托变量,就像刚才例子里面的那样
  2. 使用 invoke() ,简单就直接上代码例子:

委托的用处(配有代码例子)

  • 回调:

委托是实现回调的基础,它允许程序将一个方法(或函数)作为参数传递给另一个方法,然后在某个事件发生时,由接收方法调用这个方法

using System;namespace 练习
{// 定义一个名为Button的类,用于模拟一个按钮的行为class Button{// 定义一个委托类型MyButtonpublic delegate void MyButton();// ClickButton是一个委托类型的成员变量,用于存储按钮点击时的回调方法public MyButton ClickButton;public void Click(){Console.WriteLine("准备点击按钮");if (ClickButton != null){ClickButton();}}}internal class Program{static void MyClick(){Console.WriteLine("按钮被点击了!");}static void Main(string[] args){// 创建Button类的一个实例Button button = new Button();            // 将ClickButton委托设置为引用MyClick方法,这样点击按钮时会调用MyClickbutton.ClickButton = new Button.MyButton(MyClick);           // 模拟点击按钮,这将触发ClickButton委托,调用MyClick方法button.Click();         Console.Read();}}
}
  • 事件处理

事件是基于委托实现的。事件允许对象在发生特定行为时通知其他对象,而委托提供了一种机制来注册和触发这些通知

  • 多播委托

即一个委托可以引用多个方法。这使得你可以将多个方法链接在一起,以便它们可以依次或同时被调用。委托对象可使用 "+" 运算符进行合并。一个合并委托调用它所合并的两个委托。只有相同类型的委托可被合并。"-" 运算符可用于从合并的委托中移除组件委托。

using System;namespace 委托的多播
{internal class Program{// 定义一个委托类型CalculateDelegate,它可以引用接受两个int参数并返回int的方法public delegate int CalculateDelegate(int x, int y);// Add方法实现了CalculateDelegate委托的签名,用于加法运算public static int Add(int x, int y){// 输出加法运算的结果Console.WriteLine(x + y);return x + y;}// Multiply方法实现了CalculateDelegate委托的签名,用于乘法运算public static int Multiply(int x, int y){// 输出乘法运算的结果Console.WriteLine(x * y);return x * y;}static void Main(string[] args){// 创建CalculateDelegate委托的实例cal0,引用Add方法CalculateDelegate cal0 = new CalculateDelegate(Add);// 创建CalculateDelegate委托的实例cal1,引用Multiply方法CalculateDelegate cal1 = new CalculateDelegate(Multiply);// 通过委托的加法运算创建一个多播委托cal,它将依次执行cal0和cal1引用的方法CalculateDelegate cal = cal0 + cal1;// 调用多播委托cal,传入1和3作为参数// 这将依次调用Add和Multiply方法,并输出结果cal(1, 3);Console.ReadKey();}}
}
  • 泛型委托

以引用具有泛型参数的方法,从而提供类型安全和灵活性

using System;// 定义一个泛型委托
public delegate R MyFunc<T, R>(T t);class Program
{// 一个接受整数并返回字符串的方法static string ConvertToString(int number){return number.ToString();}static void Main(){// 创建泛型委托的实例,引用一个方法MyFunc<int, string> toString = ConvertToString;// 使用委托调用方法string result = toString(123);Console.WriteLine(result); // 输出 "123"// 可以引用不同类型的方法MyFunc<double, int> round = Math.Round;int roundedResult = round(3.14159); // 调用 Math.Round 方法Console.WriteLine(roundedResult); // 输出 "3"}
}
  • 法的抽象表示

委托允许你将方法视为一个对象,这使得方法可以像其他对象一样被传递、存储和操作

using System;namespace MethodAbstractionExample
{// 定义一个委托类型,它抽象表示一个方法,该方法接受两个int参数并返回一个int结果public delegate int Operation(int x, int y);class Program{// 这是一个具体的加法方法,符合委托的签名public static int Add(int a, int b){return a + b;}// 这是一个具体的减法方法,也符合委托的签名public static int Subtract(int a, int b){return a - b;}static void Main(string[] args){// 创建委托实例,将加法方法作为抽象方法引用Operation operation = new Operation(Add);int result = operation(5, 3); // 调用抽象方法,实际执行加法Console.WriteLine($"结果是: {result}"); // 输出 8// 可以改变委托引用的方法,以抽象表示不同的方法operation = new Operation(Subtract);result = operation(5, 3); // 这次实际执行的是减法Console.WriteLine($"结果是: {result}"); // 输出 2// 等待用户输入,以便查看控制台输出Console.ReadLine();}}
}
  • 匿名方法和 Lambda 表达式

public delegate void MyDelegate(string message);public class Program
{public static void Main(){MyDelegate del = delegate (string msg){Console.WriteLine(msg);};del("!");}
}
public delegate void MyDelegate(string message);public class Program
{public static void Main(){MyDelegate del = (msg) => Console.WriteLine(msg);del("!");}
}

为什么使用委托

委托是一种引用类型,表示对具有特定参数列表和返回类型的方法的引用。与 C 或 C++ 中的函数指针不同,委托是类型安全的,并且是面向对象的。委托不仅能够实现代码的重用,还能促进逻辑的解耦,从而提高代码的可维护性和可扩展性。


其他委托形式

Action

  • Action 表示无参数且无返回值的委托。
  • Action<int, string> 是一个带有参数且无返回值的泛型委托,最多支持16个参数。
  • Action 可以涵盖所有无返回值的委托,是对无返回值委托的一个统一封装。
using System;public class Program
{static void DisplayMessage(){Console.WriteLine("Hello, World!");}static void DisplayMessage(string message){Console.WriteLine(message);}public static void Main(){// 无参数 ActionAction actionWithoutParams = DisplayMessage;actionWithoutParams(); // 输出 "Hello, World!"// 带参数 ActionAction<string> actionWithParams = DisplayMessage;actionWithParams("Hello!"); // 输出 "Hello!"}
}

Func

  • Func 是一种表示有返回值的委托,它必须返回一个值。
  • Func 可以无参数,也可以接受最多16个参数,其中最后一个参数指定返回值的类型。
  • Func 的使用方式与 delegate 相同,但它专门用于处理有返回值的情况,为这些委托提供了一个统一的封装。
using System;public class Program
{// Func 示例,接受两个int参数,返回它们的和static int Sum(int x, int y){return x + y;}public static void Main(){// 创建 Func 委托实例Func<int, int, int> sumFunc = Sum;// 调用 Func 委托int result = sumFunc(5, 10); // 返回 15Console.WriteLine($"result: {result}");}
}

 Predicate<T>

用于表示接受一个类型为 T 的参数并返回 bool 的方法。我们展示了如何使用筛选数组中的偶数

using System;
using System.Linq;public class Program
{public static void Main(){// 创建一个整数数组int[] numbers = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };// 使用 Predicate<T> 来筛选数组中的偶数Predicate<int> isEven = number => number % 2 == 0;// 筛选出所有偶数int[] evenNumbers = numbers.Where(isEven).ToArray();// 输出筛选结果Console.WriteLine("numbers:");foreach (var number in evenNumbers){Console.WriteLine(number);}}
}

这篇关于C# 委托详解(Delegate)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1115839

相关文章

一文深入详解Python的secrets模块

《一文深入详解Python的secrets模块》在构建涉及用户身份认证、权限管理、加密通信等系统时,开发者最不能忽视的一个问题就是“安全性”,Python在3.6版本中引入了专门面向安全用途的secr... 目录引言一、背景与动机:为什么需要 secrets 模块?二、secrets 模块的核心功能1. 基

一文详解MySQL如何设置自动备份任务

《一文详解MySQL如何设置自动备份任务》设置自动备份任务可以确保你的数据库定期备份,防止数据丢失,下面我们就来详细介绍一下如何使用Bash脚本和Cron任务在Linux系统上设置MySQL数据库的自... 目录1. 编写备份脚本1.1 创建并编辑备份脚本1.2 给予脚本执行权限2. 设置 Cron 任务2

一文详解如何在idea中快速搭建一个Spring Boot项目

《一文详解如何在idea中快速搭建一个SpringBoot项目》IntelliJIDEA作为Java开发者的‌首选IDE‌,深度集成SpringBoot支持,可一键生成项目骨架、智能配置依赖,这篇文... 目录前言1、创建项目名称2、勾选需要的依赖3、在setting中检查maven4、编写数据源5、开启热

Python常用命令提示符使用方法详解

《Python常用命令提示符使用方法详解》在学习python的过程中,我们需要用到命令提示符(CMD)进行环境的配置,:本文主要介绍Python常用命令提示符使用方法的相关资料,文中通过代码介绍的... 目录一、python环境基础命令【Windows】1、检查Python是否安装2、 查看Python的安

HTML5 搜索框Search Box详解

《HTML5搜索框SearchBox详解》HTML5的搜索框是一个强大的工具,能够有效提升用户体验,通过结合自动补全功能和适当的样式,可以创建出既美观又实用的搜索界面,这篇文章给大家介绍HTML5... html5 搜索框(Search Box)详解搜索框是一个用于输入查询内容的控件,通常用于网站或应用程

Python中使用uv创建环境及原理举例详解

《Python中使用uv创建环境及原理举例详解》uv是Astral团队开发的高性能Python工具,整合包管理、虚拟环境、Python版本控制等功能,:本文主要介绍Python中使用uv创建环境及... 目录一、uv工具简介核心特点:二、安装uv1. 通过pip安装2. 通过脚本安装验证安装:配置镜像源(可

C++ 函数 strftime 和时间格式示例详解

《C++函数strftime和时间格式示例详解》strftime是C/C++标准库中用于格式化日期和时间的函数,定义在ctime头文件中,它将tm结构体中的时间信息转换为指定格式的字符串,是处理... 目录C++ 函数 strftipythonme 详解一、函数原型二、功能描述三、格式字符串说明四、返回值五

LiteFlow轻量级工作流引擎使用示例详解

《LiteFlow轻量级工作流引擎使用示例详解》:本文主要介绍LiteFlow是一个灵活、简洁且轻量的工作流引擎,适合用于中小型项目和微服务架构中的流程编排,本文给大家介绍LiteFlow轻量级工... 目录1. LiteFlow 主要特点2. 工作流定义方式3. LiteFlow 流程示例4. LiteF

CSS3中的字体及相关属性详解

《CSS3中的字体及相关属性详解》:本文主要介绍了CSS3中的字体及相关属性,详细内容请阅读本文,希望能对你有所帮助... 字体网页字体的三个来源:用户机器上安装的字体,放心使用。保存在第三方网站上的字体,例如Typekit和Google,可以link标签链接到你的页面上。保存在你自己Web服务器上的字

MySQL存储过程之循环遍历查询的结果集详解

《MySQL存储过程之循环遍历查询的结果集详解》:本文主要介绍MySQL存储过程之循环遍历查询的结果集,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言1. 表结构2. 存储过程3. 关于存储过程的SQL补充总结前言近来碰到这样一个问题:在生产上导入的数据发现