您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > C/C++/C#

C#语法——委托,架构师必修

时间:2019-12-23 10:57:26  来源:  作者:

C#语法——委托,架构师必修

C#语法——委托,架构的血液

本篇文章主要介绍委托的应用。

委托是大家最常见的语法了,但会用与精通之间的差别是巨大的。

一个程序员如果不能精通委托,那么,他永远无法成为一个架构师。

所以,委托是必须掌握的技能之一。

委托的定义

什么是委托?

委托实际上是一种类型,是一种引用类型。

微软用delegate关键字来声明委托,delegate与int,string,double等关键字一样。都是声明用的。

下面先看下声明代码,这里声明了两个委托。

public delegate void TestDelegate(string message);
public delegate int TestDelegate(MyType m, long num);

delegate既然是关键字,和int,string一样,那么,为什么delegate后又跟了一个void或者int呢?

如果他们是同等地位的关键字,为什么可以一起使用呢?

很简单,我们把delegate后面的 【void TestDelegate(string message)】理解为一个变量,是不是就清晰明了了一些。

我们把delegate关键字理解为,是用来专门来定义这种复杂的变量的。而这种复杂的变量可以包含一个返回值和任意数目任意类型的传入参数。

有没有感觉,这个复杂的变量特别像一个函数的定义。

没错,官方定义,委托类型的声明与方法签名相似。所以,这个复杂变量,的确,书写的方式就是与函数一样。

那么,为什么这个声明方式如此怪异呢,是因为,我们用delegate定义的变量,只能用函数赋值。赋值方式如下所示:

public delegate void TestDelegate(string message);
public delegate long TestDelegate2(int m, long num);
public static void Excute()
{
 TestDelegate2 td = Double; 
} 
static long Double(int m, long num)
{
 return m * num;
}

委托的基本应用

学会了赋值以后,我开始使用委托。

委托的使用方式如下:

public static void Excute()
{
TestDelegate2 td = Double; 
string result = td(51, 8);
Console.WriteLine(result);
} 

这里我们会发现,委托的使用方式与函数调用一样。

没错,它们的确是一样的。因为委托是用函数来赋值的,所以调用方式一样也并不奇怪,不是吗。

换一种说法,就是委托封装了一个函数。

如果委托是封装的函数,并且它又是引用类型。那么委托第一种常规的应用就浮现出来了。

那就是——引用类型的函数。

如果函数是引用类型,那么这个函数只要没被内存回收,就可以被调用。如果是public函数或者是public static函数,那么它能跨越的东西就更多了。

比如可以跨类调用,跨程序集调用等等。而这种用法,就是委托的基本应用。

匿名委托的应用

匿名委托的官方介绍:在 2.0 之前的 C# 版本中,声明委托的唯一方式是使用命名方法。 C# 2.0 引入匿名方法,在 C# 3.0 及更高版本中,Lambda 表达式取代匿名方法作为编写内联代码的首选方式。

看不懂没关系,我们直接来学习使用。代码如下:

delegate string anonymousDelegate(int m, long num);
public static void Excute()
{
 anonymousDelegate ad = delegate (int m, long num) { return m.ToString() + num.ToString(); };//2.0时代的匿名委托

 anonymousDelegate ad2 = (m, num) => { return m.ToString() + num.ToString(); };//3.0以后匿名委托 
}

如代码所示,匿名委托也叫Lambda表达式,不懂的同学就当它是有固定写法即可,不用讲什么道理,只要记住并应用即可。

匿名委托虽然减少了一点代码,但还是要求我们自己去声明委托。所有,还能再简写一点吗?

答案当然是,可以的。

Action与Func

Action与Func是微软为我们预先定义好了的,两个委托变量。其中Action是不带返回值的委托,Func是带返回值的委托。

可以说,Action与Func完全包含了,我们日常使用所需的,全部的,委托变量。

也就是说,我们可以不用再去自己手动声明委托了。

下面来看最简单的Action与Func的定义:

Action a1 = () => { };
Func<int> f1 = () => { return 1; };//必须写 return 1;

Action与Func是泛型委托,最多支持16个入参变量。下面代码为一个入参的定义,多参数以此类推。

Action<int> a1 = (i) => { };
Func<string,int> f1 = (str) => { return 1;//必须写 return 1; };

委托的线程应用

委托的线程应用是委托的第二种用法,分为线程使用委托,和委托的异步应用两种。

我们先看线程使用委托。如下代码所示,一个无入参匿名Action和一个无入参匿名Func。

Task taskAction = new Task(() => { });//无入参匿名Action
taskAction.Start(); 

Task<int> taskFunc = new Task<int>(() => { return 1; });//无入参匿名Func
taskFunc.Start();

int result= taskFunc.GetAwaiter().GetResult();//获取线程返回结果

我们能看到两种委托应用,代码都非常简洁。

下面我们再来看委托的异步应用。首先看最简单的异步调用。

Action action = new Action(() => { });
IAsyncResult result = action.BeginInvoke((iar) =>
{
}, null);
 
Func<int> func = new Func<int>(() => { return 1; }); 
IAsyncResult resultfunc = func.BeginInvoke((iar) =>
{
 var res = func.EndInvoke(iar); 
}, null);

这里我们使用委托的BeginInvoke方法来开启线程,进行异步调用。如上面代码所示,这里介绍了Action与Func的最基础的异步应用。

委托是架构的血液,如果系统中没有委托,那代码将堆叠到一起,耦合性就很高。

就好比一碗汤面倒掉了所有的汤,只要它静放一个阵子,就会变成一坨面球,让你无从下嘴。

所以,委托是是框架的流畅的基石。

那么委托到底是如何流动的呢?

我们先从刚介绍过的委托的线程应用说起。

第一核心应用——随手线程:

我们在做开发的时候,一定接触过父类。父类是干什么的呢?父类通常是用来编写公共属性和函数,方便子类调用的。

那我们的委托的第一个核心应用,就是父类的公共函数,线程随手启动。如何随手开启呢?

首先,我们创建父类代码如下:

class BaseDelegateSyntax
{ 
 public void AsyncLoad(Action action)
 {
 
 }
 public void AsyncLoad(Action action, Action callback)
 {
 IAsyncResult result = action.BeginInvoke((iar) =>
 {
 callback();
 }, null);
 }

 public void AsyncLoad<T>(Action<T> action, T para, Action callback)
 {
 IAsyncResult result = action.BeginInvoke(para, (iar) =>
 {
 callback();
 }, null);
 }

 public void AsyncLoad<T, R>(Func<T, R> action, T para, Action<R> callback)
 {
 IAsyncResult result = action.BeginInvoke(para, (iar) =>
 {
 var res = action.EndInvoke(iar);
 callback(res);
 }, null);
 }
}

我们看到上面的代码,父类中添加了四个异步委托的调用函数,接下来,我们就可以在继承该类的子类中,随手开启线程了。

子类代码如下:

class ChildDelegateSyntax : BaseDelegateSyntax
{
 public void Excute()
 {
 //开启异步方法
 base.AsyncLoad(() => { });
 
 //开启异步方法,并且在异步结束后,触发回调方法
 base.AsyncLoad(() => { },
 ()=> 
 {
 //我是回调方法
 });
 
 //开启异步有入参的方法,传递参数,并且在异步结束后,触发回调方法
 base.AsyncLoad<string>((s) => { },"Kiba518",
 () =>
 {
 //我是回调方法
 });
 
 //开启异步有入参的方法,传递字符串参数Kiba518,之后返回int型结果518,
 //并且在异步结束后,触发回调方法,回调函数中可以获得结果518
 base.AsyncLoad<string,int>((s) => {
 return 518;
 }, "Kiba518",
 (result) =>
 {
 //我是回调方法 result是返回值518
 });
 }
}

看了上面的父子类后,是否感觉委托让我们繁杂的线程世界变简洁了呢?

第二核心应用——穿越你的世界:

接下来,我们来看委托的第二种核心用法,穿越的应用。

这个应用,是最常见,也最普通的应用了。因为委托是引用类型,所以A类里定义的委托,可以在被内存回收之前,被其他类调用。

我们经常会在各种论坛看到有人发问,A页面如何调用B页面的属性、方法、父页面获取子页面的属性、方法,或者子页面获取父页面的属性、方法。

其实,只要定义好委托,并将委托正确的传递,就可以实现穿越的调用了。

下面我们看下穿越应用的代码。

public class FirstDelegateSyntax
{
 public FirstDelegateSyntax()
 {
 Console.WriteLine(" First 开始 " );
 SecondDelegateSyntax sds = new SecondDelegateSyntax(()=> {
 Console.WriteLine(" First传给Second委托被触发 ");
 });
 sds.Excute();
 Console.WriteLine(" First 结束 ");
 }
}
 
public class SecondDelegateSyntax
{
 public Action Action { get; set; }
 public SecondDelegateSyntax(Action _action)
 {
 Console.WriteLine(" Second的构造函数 ");
 Action = _action;
 }
 public void Excute()
 {
 Console.WriteLine(" Second的Excute被触发 ");
 Action();
 }
}

我们可以看到,我们传递的委托,穿越了自身所属的类。在SecondDelegateSyntax类中被触发了。

运行结果如下:

C#语法——委托,架构师必修

 

第三核心应用——回调函数:

世界上本没有回调函数,叫的人多了,也就有了。

请记住,所有的回调函数,都是委托的穿越应用,所有的回调函数;都是委托的穿越应用;所有的回调函数,都是委托的穿越应用。

重要的话要讲三遍。

因为委托是引用类型,所以可以被[址传递]。函数是不可以被传递的。

当你传递函数的时候,其实是匿名传递了一个委托的地址。

结语

委托是我们最常用的语法,它将函数封装成引用类型的变量,供其他单位调用。

因为委托的特质是引用类型,所以决定了委托是可以进行址传递。也就是说,委托是穿梭于我们系统代码中的列车。

我们可以在列车上放很多很多东西,在需要的站点,叫停列车,并将托运的东西搬下来使用。

所以,理论上,只要我们利用好委托,就可以大量减少冗余的代码。

但委托这种列车,是每个程序员都可以定义的,如果一个项目中有十个开发者,每个人都在定义委托,那么,就有可能出现定义了十个相同的委托的情况,这样就出现了撞车的现象。

所以委托在使用的时候,尽量做到有序传递,即预先做好列车的行驶路线,让委托按照路径运行,尽量不要定义可以被任何单位调用的公共委托。

 



Tags:C#   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
一、简介很多时候我们都需要用到一些验证的方法,有时候需要用正则表达式校验数据时,往往需要到网上找很久,结果找到的还不是很符合自己想要的。所以我把自己整理的校验帮助类分...【详细内容】
2021-12-27  Tags: C#  点击:(2)  评论:(0)  加入收藏
读取SQLite数据库,就是读取一个路径\\192.168.100.**\position\db.sqlite下的文件<startup useLegacyV2RuntimeActivationPolicy="true"> <supportedRuntime version="v4.0"/...【详细内容】
2021-12-16  Tags: C#  点击:(21)  评论:(0)  加入收藏
一文看懂"async"和“await”关键词是如何简化了C#中多线程的开发过程当我们使用需要长时间运行的方法(即,用于读取大文件或从网络下载大量资源)时,在同步的应用程序中,应用程序本...【详细内容】
2021-12-01  Tags: C#  点击:(24)  评论:(0)  加入收藏
OpenCV(Open Source Computer Vision Library)是一个(开源免费)发行的跨平台计算机视觉库,可以运行在Linux、Windows、Android、ios等操作系统上,它轻量级而且高效---由一系列...【详细内容】
2021-11-11  Tags: C#  点击:(50)  评论:(0)  加入收藏
在进行C#接口开发时经常需要将数据集转换成json返回,下面介绍一种将DataTable数据转换成实体类,然后再转换成json的方法。一、创建全病案类/// <summary> /// 全病案 //...【详细内容】
2021-11-04  Tags: C#  点击:(45)  评论:(0)  加入收藏
C# 获取网络路径文件大小信息的方法private void button1_Click(object sender, EventArgs e){label2.Text= GetInfo(textBox1.Text);}private string GetInfo(string URL){...【详细内容】
2021-10-29  Tags: C#  点击:(52)  评论:(0)  加入收藏
现如今二维码广泛应用,已经成为了生活中不可或缺的东西;所以我们日常的开发中,也少不了对二维码的生成或者解析。此文讲述一下使用C# 生成和解析二维码 实现功能:将字符串生成二...【详细内容】
2021-10-21  Tags: C#  点击:(59)  评论:(0)  加入收藏
最开始接触.net大概是2003年,当时用的还是framework1.1,那是CAM项目,使用.net主要是利用RemotingService做分布式开发,我当时做其中的一个子系统,因为之前一直都是C++开发,对C#一...【详细内容】
2021-08-17  Tags: C#  点击:(102)  评论:(0)  加入收藏
最近想自己做个录屏工具玩玩,使用的是ffmpeg,录屏是没有问题,在取麦克风及电脑声卡遇到了难题,ffmpeg也可以获取到设备名称但是没办法解析 ( 指令:ffmpeg -list_devices true -f d...【详细内容】
2021-08-04  Tags: C#  点击:(277)  评论:(0)  加入收藏
在实际的项目开发中,可能需要考虑代码的安全性或执行效率,需要将具体的功能实现隐藏,dll接口提供了很好的解决方案。dll接口不仅在C++的项目中可以直接调用,在C#开发的项目也同...【详细内容】
2021-07-09  Tags: C#  点击:(122)  评论:(0)  加入收藏
▌简易百科推荐
一、简介很多时候我们都需要用到一些验证的方法,有时候需要用正则表达式校验数据时,往往需要到网上找很久,结果找到的还不是很符合自己想要的。所以我把自己整理的校验帮助类分...【详细内容】
2021-12-27  中年农码工    Tags:C#   点击:(2)  评论:(0)  加入收藏
引言在学习C语言或者其他编程语言的时候,我们编写的一个程序代码,基本都是在屏幕上打印出 hello world ,开始步入编程世(深)界(坑)的。C 语言版本的 hello world 代码:#include <std...【详细内容】
2021-12-21  一起学嵌入式    Tags:C 语言   点击:(11)  评论:(0)  加入收藏
读取SQLite数据库,就是读取一个路径\\192.168.100.**\position\db.sqlite下的文件<startup useLegacyV2RuntimeActivationPolicy="true"> <supportedRuntime version="v4.0"/...【详细内容】
2021-12-16  今朝我的奋斗    Tags:c#   点击:(21)  评论:(0)  加入收藏
什么是shellshell是c语言编写的程序,它在用户和操作系统之间架起了一座桥梁,用户可以通过这个桥梁访问操作系统内核服务。 它既是一种命令语言,同时也是一种程序设计语言,你可以...【详细内容】
2021-12-16  梦回故里归来    Tags:shell脚本   点击:(18)  评论:(0)  加入收藏
一、编程语言1.根据熟悉的语言,谈谈两种语言的区别?主要浅谈下C/C++和PHP语言的区别:1)PHP弱类型语言,一种脚本语言,对数据的类型不要求过多,较多的应用于Web应用开发,现在好多互...【详细内容】
2021-12-15  linux上的码农    Tags:c/c++   点击:(17)  评论:(0)  加入收藏
1.字符串数组+初始化char s1[]="array"; //字符数组char s2[6]="array"; //数组长度=字符串长度+1,因为字符串末尾会自动添&lsquo;\0&lsquo;printf("%s,%c\n",s1,s2[2]);...【详细内容】
2021-12-08  灯-灯灯    Tags:C语言   点击:(47)  评论:(0)  加入收藏
函数调用约定(Calling Convention),是一个重要的基础概念,用来规定调用者和被调用者是如何传递参数的,既调用者如何将参数按照什么样的规范传递给被调用者。在参数传递中,有两个很...【详细内容】
2021-11-30  小智雅汇    Tags:函数   点击:(19)  评论:(0)  加入收藏
一、问题提出问题:把m个苹果放入n个盘子中,允许有的盘子为空,共有多少种方法?注:5,1,1和1 5 1属同一种方法m,n均小于10二、算法分析设f(m,n) 为m个苹果,n个盘子的放法数目,则先对...【详细内容】
2021-11-17  C语言编程    Tags:C语言   点击:(49)  评论:(0)  加入收藏
一、为什么需要使用内存池在C/C++中我们通常使用malloc,free或new,delete来动态分配内存。一方面,因为这些函数涉及到了系统调用,所以频繁的调用必然会导致程序性能的损耗;另一...【详细内容】
2021-11-17  深度Linux    Tags:C++   点击:(38)  评论:(0)  加入收藏
OpenCV(Open Source Computer Vision Library)是一个(开源免费)发行的跨平台计算机视觉库,可以运行在Linux、Windows、Android、ios等操作系统上,它轻量级而且高效---由一系列...【详细内容】
2021-11-11  zls315    Tags:C#   点击:(50)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条