[C#] C# 知识回顾 - 委托 delegate
C# 知识回顾 - 委托 delegate
【博主】反骨仔 【原文】http://www.cnblogs.com/liqingwen/p/6031892.html
目录
What's 委托
delegate 一种自定义的引用类型,它包含了特定的参数列表和返回类型。
使用委托时,只需要对应的方法的签名和返回类型兼容即可,无论是实例方法,抑或是静态方法。通过调用委托的实例就相当于调用方法本身,因为委托存储的是一个方法列表,调用委托的实例就相当于依次调用方法列表的内容。委托它将方法作为参数进行传递给了其它方法,我们常用的事件处理程序就是通过委托调用的方法,也是一种观察者模式的体现。
下面的示例演示了一个委托声明:
public delegate int Del(int x, int y);
使用委托的要求是:方法签名与返回类型兼容。可以是静态方法,也可以是实例方法。
委托的特点
类型安全,类似于 C 和 C++ 中的函数指针。
可将方法作为参数进行传递。
可用于定义回调方法。
委托可以链接在一起;例如,可以对一个事件调用多个方法。
方法不必与委托类型完全匹配。
使用委托
委托,一种类型,它是安全的,自定义的,委托的名称就决定了这个委托是什么类型。
//该委托可以封装 “,参数类型 string,返回类型 void” 的方法
public delegate void MyDel(string message);
委托的实例对象通常使用两种方式进行构建,直接使用类的方法名,或者使用 Lambda 表达式,当然匿名方法也可以。
在调用委托的时刻,我们将传递到委托的参数会继续传递到委托列表的方法中。如果委托列表中包含返回值的话,会将最后一个返回值返回给调用方。也就是该委托对象调用完毕的返回值。
//该委托名为 MyDel,可以封装 “参数类型 string,返回值类型 void” 的方法
public delegate void MyDel(string message); class Program
{
static void Main(string[] args)
{
//实例化委托
MyDel del = Print;
//调用委托
del("Hi"); Console.Read();
} /// <summary>
/// 打印文本
/// </summary>
/// <remarks>这是一个可用于 MyDel 委托的方法</remarks>
/// <param name="message"></param>
private static void Print(string message)
{
Console.WriteLine(message);
}
}
委托的关键字是 delegate,它派生自 Delegate 类,也是 sealed,即密封类,不能作为基类再继续派生。
异步回调:允许以方法的形式作为参数形式进行传递,并在稍后进行该委托的调用。通过这个形式使用的委托,调用方不需要知道方法的具体实现,只是简单的把它当做一个功能即可,这类似接口的封装。
class Program
{
static void Main(string[] args)
{
MyDel del = Print;
CallbackMethod(, , del); //将委托传递到 CallbackMethod 方法 Console.Read();
} /// <summary>
/// 回调方法
/// </summary>
/// <param name="m"></param>
/// <param name="n"></param>
/// <param name="del"></param>
private static void CallbackMethod(int m, int n, MyDel del)
{
del((m + n).ToString());
} private static void Print(string message)
{
Console.WriteLine(message);
}
}
在这里的 CallbackMethod 作用是,调用委托,因为它包含的是 Print() 方法的调用,所以只需要传递对应的 string 类型作为参数即可。
我们在创建委托的时候,你可以选择使用的是实例方法或者是静态方法。当你使用的是实例方法时,该委托对象会同时引用该实例的对象及它的方法。委托并不关心应用引用对象的类型,它关心的是,方法签名和返回值兼容,即可。不过,如果你创建委托对象包含的是静态方法的时候,它是只引用该方法的。
使用 += 可以把多个方法添加到一个委托对象的调用列表中,调用一次委托,相当于一次性调用一堆方法。
//该委托可以封装 “名 MyDel,参数类型 string,返回值类型 void” 的方法
public delegate void MyDel(string message); class MyClass
{
public void Print1(string message)
{
Console.WriteLine($"{message} - {nameof(Print1)}");
} public void Print2(string message)
{
Console.WriteLine($"{message} - {nameof(Print2)}");
}
} class Program
{
static void Main(string[] args)
{
var myClass = new MyClass();
MyDel del1 = myClass.Print1;
MyDel del2 = myClass.Print2;
MyDel del3 = Print; var del = del1 + del2;
del += del3; //这里使用 +=
del("Hi!"); Console.Read();
} private static void Print(string message)
{
Console.WriteLine($"{message} - {nameof(Print)}");
}
}
委托对象 del,他内部存储的是一个包含三个方法的调用列表(Print1、Print2 和 Print),在你调用 del 对象时,调用列表中的方法会依次调用。
多播委托:一个委托对象调用多个方法,使用 +=。
若要从委托对象的调用列表中移除方法,需要使用 -=。
static void Main(string[] args)
{
var myClass = new MyClass();
MyDel del1 = myClass.Print1;
MyDel del2 = myClass.Print2;
MyDel del3 = Print; var del = del1 + del2;
del += del3; //使用 +=
del("Hi!"); Console.WriteLine("======分割线======"); del -= del2; //使用 -=
del("Hi!"); Console.Read();
}
你也可以编写一些方法获取调用列表中方法的数量:
static void Main(string[] args)
{
var myClass = new MyClass();
MyDel del1 = myClass.Print1;
MyDel del2 = myClass.Print2;
MyDel del3 = Print; var del = del1 + del2;
del += del3; //使用 +=
//del("Hi!"); var count = del.GetInvocationList().Length; //获取委托调用列表中方法的数量
Console.WriteLine(count); Console.WriteLine("======分割线======"); del -= del2; //使用 -=
//del("Hi!"); count = del.GetInvocationList().Length; //获取委托调用列表中方法的数量
Console.WriteLine(count); Console.Read();
}
多播委托派生自 MulticastDelegate,也是继承自 Delegate的,常用于事件处理中。
传送门
【参考】https://msdn.microsoft.com/zh-cn/library/windows/apps/ms173171(v=vs.120).aspx
【参考】微软官方文档
[C#] C# 知识回顾 - 委托 delegate的更多相关文章
- [.NET] C# 知识回顾 - 委托 delegate (续)
C# 知识回顾 - 委托 delegate (续) [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6046171.html 序 上篇<C# 知识回 ...
- C#基础知识回顾--委托事件
在上一篇看到他我一下子就悟了(续)---委托,被人狂喷.说写的太空,没有什么内容之类的.所以准备在这里重写下,不过还是按着以前的方式尽量简单的写.这里我们以打篮球为例. 第一步:首先,其他对象订购事件 ...
- [C#] C# 知识回顾 - 你真的懂异常(Exception)吗?
你真的懂异常(Exception)吗? 目录 异常介绍 异常的特点 怎样使用异常 处理异常的 try-catch-finally 捕获异常的 Catch 块 释放资源的 Finally 块 一.异常介 ...
- [C#] C# 知识回顾 - 学会处理异常
学会处理异常 你可以使用 try 块来对你觉得可能会出现异常的代码进行分区. 其中,与之关联的 catch 块可用于处理任何异常情况. 一个包含代码的 finally 块,无论 try 块中是否在运行 ...
- [C#] C# 知识回顾 - 学会使用异常
学会使用异常 在 C# 中,程序中在运行时出现的错误,会不断在程序中进行传播,这种机制称为“异常”. 异常通常由错误的代码引发,并由能够更正错误的代码进行 catch. 异常可由 .NET 的 CLR ...
- [C#] C# 知识回顾 - 异常介绍
异常介绍 我们平时在写程序时,无意中(或技术不够),而导致程序运行时出现意外(或异常),对于这个问题, C# 有专门的异常处理程序. 异常处理所涉及到的关键字有 try.catch 和 finally ...
- [.NET] C# 知识回顾 - Event 事件
C# 知识回顾 - Event 事件 [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6060297.html 序 昨天,通过<C# 知识回顾 - ...
- [.NET] C# 知识回顾 - 事件入门
C# 知识回顾 - 事件入门 [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6057301.html 序 之前通过<C# 知识回顾 - 委托 de ...
- C# 知识回顾 - 装箱与拆箱
装箱与拆箱 目录 生活中的装箱与拆箱 C# 的装箱与拆箱 值类型和引用类型 装箱 拆箱 生活中的装箱与拆箱 我们习惯了在网上购物,这次你想买本编程书 -- <C 语言从入门到放弃> ...
随机推荐
- 前端开发中SEO的十二条总结
一. 合理使用title, description, keywords二. 合理使用h1 - h6, h1标签的权重很高, 注意使用频率三. 列表代码使用ul, 重要文字使用strong标签四. 图片 ...
- ASP.NET Core应用的错误处理[2]:DeveloperExceptionPageMiddleware中间件如何呈现“开发者异常页面”
在<ASP.NET Core应用的错误处理[1]:三种呈现错误页面的方式>中,我们通过几个简单的实例演示了如何呈现一个错误页面,这些错误页面的呈现分别由三个对应的中间件来完成,接下来我们将 ...
- Java MyBatis 插入数据库返回主键
最近在搞一个电商系统中由于业务需求,需要在插入一条产品信息后返回产品Id,刚开始遇到一些坑,这里做下笔记,以防今后忘记. 类似下面这段代码一样获取插入后的主键 User user = new User ...
- 如何一步一步用DDD设计一个电商网站(一)—— 先理解核心概念
一.前言 DDD(领域驱动设计)的一些介绍网上资料很多,这里就不继续描述了.自己使用领域驱动设计摸滚打爬也有2年多的时间,出于对知识的总结和分享,也是对自我理解的一个公开检验,介于博客园这个平 ...
- CRL快速开发框架系列教程五(使用缓存)
本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...
- 脑洞大开之采用HTML5+SignalR2.0(.Net)实现原生Web视频
目录 对SignalR不了解的人可以直接移步下面的目录 SignalR系列目录 前言 - -,我又来了,今天废话不多说,我们直接来实现Web视频聊天. 采用的技术如下: HTML5 WebRTC Si ...
- log4net使用手册
1. log4net简介 log4net是.Net下一个非常优秀的开源日志记录组件.log4net记录日志的功能非常强大.它可以将日志分不同的等级,以不同的格式,输出到不同的媒介.Java平台下,它还 ...
- 升级npm
查看npm的所有版本 运行命令: npm view npm versions 命令运行后,会输出到目前为止npm的所有版本. [ '1.1.25', '1.1.70', '1.1.71', '1.2. ...
- BPM助力企业数字化转型
自九十年代末,流程管理开始引入国内,至今已经有20多年的历史了,由最初的部门级应用向企业级应用转变,大家的认知也经历了一系列的发展变化.不同阶段的信息化水平对企业的流程以及BPM平台也提出了不同的需求 ...
- ios label 自动计算行高详解
在OC当中自动计算行高主要调用系统的 p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 11.0px Menlo; color: #ffffff } span ...