委托的前世今生

Posted 常哥说编程

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了委托的前世今生相关的知识,希望对你有一定的参考价值。

-Begin-

起因

很多C#初学者,都遇到过这样的问题——线程间操作无效,从不是创建控件的线程访问它。

今天就这个问题,展开分析。

 

 

溯源

先说下这个问题产生的根源。

大家都知道,程序运行起来之后,首先会有一个主线程,主线程用于处理控件生成、界面渲染、事件响应、逻辑处理等操作,因此我们可以理解为窗体里的控件是属于主线程的。

我们也知道多线程,如果我们想实现与主线程同时执行另一件事,一般会去使用多线程。

因此多线程,从某种意义上来说,它和主线程都属于“线程”这个家族,他们的身份是“平等”的,就像你和你同事之间的关系一样。

那么,试想一下,如果你同事有一天想从你手上把你的PLC或者上位机项目程序拿过去,你愿不愿意?

 

 所以,如果在多线程里操作主线程的控件,你觉得主线程会不会答应,当然不会,主线程不答应的最直接表现就是,它会直接给一个报错,权当警告,如下图所示:

 

 

解决

那么如何解决呢?

你的同事如果非要拿到你的程序,他会想,硬的不行,就来软的。

所以他会找到你们共同的领导,跟你们领导这样说:“我手头上的这个100万的项目,能给公司带来50%的利润,现在需要用到他之前那个项目里的一个小知识,需要他把程序给我参考一下”。

在公司利益面前,你觉得你的领导会怎么办?

于是,领导和你“商量”了一下,毋庸置疑,你妥协了。

你的同事使用的招数叫做——委托。

那么,现在回到之前的问题上来,现在多线程要操作主线程的控件,是不是也可以使用委托来实现?

前世

委托定义:委托(Delegate) 是对某个方法的引用的一种引用类型变量。

 

如果这句话看不懂,那就别看了,跟着我动手做。

1、声明委托

委托声明需要根据执行的方法来定,严格来说,就是根据执行方法的返回值和参数,我们只是给窗体的Text设置一个固定值而已,因此我们的参数是空,返回值也为空。

声明委托如下:

 

 

2、创建委托对象

委托严格来说是一种类型,就像类一样,如果想要调用某个类,必须要创建一个该类的对象,所以我们要创建一个委托对象:

 

 

3、创建委托方法

委托对象也只是一个对象而已,就像领导一样,领导是不可能干活的,最终干活还得靠底下的兵来干,所以我们还得招人去干活。

招人干活就是委托方法,我们现在这个活很简单,所以我们的方法也很简单。

 

 

4、委托绑定

我们招到了一个“兵”,现在也有一个部门领导,怎么把他们联系起来呢?

很简单,让人事把这个兵分到这个部门就行了,这个分配的过程就是委托绑定,代码如下:

 

 

5、委托调用

万事俱备,只欠东风,终于干活了。

作为公司的老板,一般是不可能跟员工打交道的,他会把任务分配给部门领导,部门领导会把活再分配下去,所以我们委托调用,也是调用委托对象。

 

 

 

以上五步,就是委托的实现过程。

然而,我们运行之后,还是会报错。

 

 

没有那么简单的事!

因为想要在多线程里操作主线程的控件,你还得经过控件的同意,怎么经过控件同意呢?

控件的父类Control提供了一个这样的方法:

 

 意思就是说,想要操作控件,必须要通过Invoke方法来实现,Invoke方法里参数是一个委托,于是,我们只能灰溜溜地,这样写:

 

 果然,按照规矩来,就能达到效果:

 

 

今生

微软从某个版本开始,出来了Action和Lamda表达式,Action是系统委托,也就是说,不需要我们手动创建委托了,它有个兄弟叫Func,Action没有返回值,最多可以有16个参数,Func必须要有返回值,最多可以有16个参数,最后一个参数表示返回值。

于是我们开始简化:

第一步简化:用Action作为委托来创建

 

 第二步简化:委托对象只用一次,所以可以直接放到参数里

 

 第三步简化:用Lamda表达式代替方法

 

 

 

总结

我们所以常写的那行代码,其实只是一种简写方式而已,委托的五步法,不管怎么简化,怎么优化,其实本质还是一样,都离开不了这五个步骤。

这就是经典。

都看到这里了,是不是要点个赞呢?

 

以上是关于委托的前世今生的主要内容,如果未能解决你的问题,请参考以下文章

源码解析 | Dubbo-SPI和IoC的前世今生

husky的配置及前世今生

云计算的前世今生

原创 | 函数 scanf 的前世今生

GIT前世今生

.Net Core001.NetCore前世今生