[C#基础]说说委托+=和-=的那些事

写在前面

为什么会突然想说说委托?原因吗,起于一个同事的想法,昨天下班的路上一直在想这个问题,如果给委托注册多个方法,会不会都执行呢?为了一探究性,就弄了个demo研究下。

+=

大家都知道委托都继承自System.MulticastDelegate,而System.MulticastDelegate又继承自System.Delegate,可以通过+=为委托注册多个方法。那么他们是否都执行了呢?执行的结果又是怎样的呢?有返回值和没返回值的是否结果是否一样?那就试着说说+=都干了哪些事?

测试代码

 1 namespace Wolfy.DelegateDemo
 2 {
 3     public delegate void ShowMsg(string msg);
 4     public delegate int MathOperation(int a, int b);
 5     class Program
 6     {
 7         static ShowMsg showMsg;
 8         static MathOperation mathOperation;
 9         static void Main(string[] args)
10         {
11             showMsg += ShowHello;
12             showMsg += ShowHello1;
13             showMsg("大家新年好啊");
14             mathOperation += Add;
15             mathOperation += Multiply;
16             int result = mathOperation(1, 2);
17             Console.WriteLine(result.ToString());
18             Console.Read();
19         }
20         static void ShowHello(string msg)
21         {
22             Console.WriteLine("哈喽:" + msg);
23         }
24         static void ShowHello1(string msg)
25         {
26             Console.WriteLine("哈喽1:" + msg);
27         }
28         static int Add(int a, int b)
29         {
30             return a + b;
31         }
32         static int Multiply(int a, int b)
33         {
34             return a * b;
35         }
36     }
37 }

你可以猜猜运行结果,如下图:

可以看到没有返回值的都输出了,有返回值的只输出了Mutiply的结果,那么+=内部做了哪些事?可以看一下反编译的代码:

 1 using System;
 2 namespace Wolfy.DelegateDemo
 3 {
 4     internal class Program
 5     {
 6         private static ShowMsg showMsg;
 7         private static MathOperation mathOperation;
 8         private static void Main(string[] args)
 9         {
10             Program.showMsg = (ShowMsg)Delegate.Combine(Program.showMsg, new ShowMsg(Program.ShowHello));
11             Program.showMsg = (ShowMsg)Delegate.Combine(Program.showMsg, new ShowMsg(Program.ShowHello1));
12             Program.showMsg("大家新年好啊");
13             Program.mathOperation = (MathOperation)Delegate.Combine(Program.mathOperation, new MathOperation(Program.Add));
14             Program.mathOperation = (MathOperation)Delegate.Combine(Program.mathOperation, new MathOperation(Program.Multiply));
15             Console.WriteLine(Program.mathOperation(1, 2).ToString());
16             Console.Read();
17         }
18         private static void ShowHello(string msg)
19         {
20             Console.WriteLine("哈喽:" + msg);
21         }
22         private static void ShowHello1(string msg)
23         {
24             Console.WriteLine("哈喽1:" + msg);
25         }
26         private static int Add(int a, int b)
27         {
28             return a + b;
29         }
30         private static int Multiply(int a, int b)
31         {
32             return a * b;
33         }
34     }
35 }

通过上面的代码可以看出+=内部是通过委托的 Combine静态方法将委托进行组合的,可以看一下委托的这个静态方法是如何实现的。

可以看到最终调用CombineImpl这个方法,这个方法内部很奇怪:

并没有我们想看到的代码,那这个方法是干嘛用的啊?

MSDN的解释

Concatenates the invocation lists of the specified multicast (combinable) delegate and the current multicast (combinable) delegate.

大概意思就是:将当前的委托加入到指定的多播委托集合中。

绕了一圈那么有返回值的委托,到底执行了么?那也只能通过调试来看看了。(绕了一圈,又回到了编辑器,唉)

继续F11你会发现确实进入了Add方法

也确实执行了,但在遍历多播委托集合的时候,将之前的值给覆盖了。

那么现在可以得出这样的结论了:无返回值的委托,你给它注册多少个方法,它就执行多少个方法,而有返回值的委托,同样注册多少个方法就执行多少个方法,但返回的是最后一个方法的返回值。

-=

既然说了+=,那么作为收拾烂摊子的-=也不得不说。在项目中使用了+=就要使用-=来释放。那它内部做了哪些事?同样使用上面的代码,在输出结果后,使用-=来释放资源。

可以看出,使用-=内部是调用了委托的Remove静态方法。

使用-=最终是将委托置为null,为null另一个意思就是空引用,这样就可以等待垃圾回收器进行回收了。

总结

这个问题虽然很基础,一个同事当时问了,就给他说了一下,在下班的路上一直在想,内部是如何实现的?就试着通过反编译的方式一探究竟。但貌似CombineImpl这个方法,给的结果不太满意。没看到具体的实现。希望对你有所帮助!

时间: 2024-12-28 14:29:24

[C#基础]说说委托+=和-=的那些事的相关文章

C# 1的核心基础之一——委托

C# 1的核心基础之一--委托 C# 1的核心基础之一--委托简单委托的构成合并和删除委托事件的简单讨论委托总结 简单委托的构成 声明委托类型: delegate void StringProcessor(string input); 为委托实例的操作找到一个恰当的方法 void PrintString(string x)完全符合要求 void PrintInteger(int x)参数类型不兼容 void PrintTwoStrings(string x, string y)参数个数不匹配 i

[.net 面向对象编程基础] (21) 委托

[.net 面向对象编程基础] (20)  委托 上节在讲到LINQ的匿名方法中说到了委托,不过比较简单,没了解清楚没关系,这节中会详细说明委托. 1.什么是委托? 学习委托,我想说,学会了就感觉简单的不能再简单了,没学过或都不愿了解的人,看着就头大,其实很简单.委托在.net面向对象编程和学习设计模式中非常重要,是学习.net面向对象编程必须要学会并掌握的. 委托从字面上理解,就是把做一些事情交给别人来帮忙完成.在C#中也可以这样理解,委托就是动态调用方法.这样说明,就很好理解了. 平时我们会

【.NET基础】--委托、事件、线程(3)

之前的两篇文章我们了解了委托和事件,本文我们看一下线程. 1,一个窗体程序,默认拥有一个线程(相当于一个商店里面,只有一个店员),这个默认的线程叫做 UI线程/主线程. 2,进程和线程的关系: A,进程,包含程序运行所需要的资源 ,在大多数情况下是指 程序.(商店:囤积要使用的资源的地方) B,线程,是在进程中能够被CPU调用的程序单元,是提供给CPU运行程序的代码片段.(商店员工:是运行程序的行动者) C,一个进程至少一个线程,每一个线程有自己专属的寄存器(栈指针.程序计数器等)但代码区是共享

C#基础系列——委托实现简单设计模式

前言:上一篇介绍了下多线程的相关知识:C#基础系列--多线程的常见用法详解,里面就提到了委托变量.这篇简单介绍下委托的使用.当然啦,园子里面很多介绍委托的文章都会说道:委托和事件的概念就像一道坎,过了这个槛的人,觉得真是太容易了,而没有过去的人每次见到委托和事件就觉得心里发慌.确实这东西就像最开始学C语言的指针一样,令人有一种很纠结的感觉,总觉得要调用一个方法直接调用就行了,为啥非要定义一个委托时执行这个方法呢.其实在C#里面很多的技术都是为了重用和简化代码而生,委托也不例外,很多使用C#多态去

C#基础学习——委托

using System; namespace Test { public delegate void TryDelegate(string name); public class SayHello{ public void SpeakChinese(string name){ Console.WriteLine("你好,"+name); } public void SpeakEnglish(string name){ Console.WriteLine("Hello,&qu

C#基础:委托

委托是C#中最为常见的内容.与类.枚举.结构.接口一样,委托也是一种类型.类是对象的抽象,而委托则可以看成是函数的抽象.一个委托代表了具有相同参数列表和返回值的所有函数.比如: delegate int GetCalculatedValueDelegate(int x, int y); 在上面的定义中,我们定义了一个委托,这个委托代表着一类函数,这些函数的第一个参数是整数型的x,第二个参数是整数型的y,而函数的返回值则是一个整数.在这里,为了描述方便,我们把这一类的函数称为具有相同签名(sign

C#基础系列——委托和设计模式(二)

此篇简单抽取了几个设计模式分别按照多态和委托的方式去实现,当然这里的重点并不是讲设计模式,而是为了使读者更好地理解委托.所以设计模式的很多细节,本篇可能会略过. 一.简单工厂模式:本篇就借助计算器的例子加以说明. 1.多态实现简单工厂模式. class Program2 { static void Main(string[] args) { //1.使用多态实现简单工厂模式 int x = 8, y = 2; var iRes1 = GetObject("+").Compute(x,

js基础-事件委托

让利用事件冒泡的原理,让自己的所触发的事件,让他的父元素代替执行! <!DOCTYPE html> <html> <head> <title>事件委托</title> </head> <body> <ul id="ul"> <li class="item">11</li> <li class="item">22<

[C#基础知识]委托详解

1. 委托是什么? 个人认为,可以从以下2点来理解:  (1) 从数据结构来讲,委托是和类一样是一种用户自定义类型. (2) 从设计模式来讲,委托(类)提供了方法(对象)的抽象. 既然委托是一种类型,那么它存储的是什么数据? 我们知道,委托是方法的抽象,它存储的就是一系列具有相同签名和返回回类型的方法的地址.调用委托的时候,委托包含的所有方法将被执行. 2. 委托类型的定义 委托是类型,就好像类是类型一样.与类一样,委托类型必须在被用来创建变量以及类型对象之前声明. delegate void