一、委托的内部实现

C#中的委托是一种类型安全的回调函数,假设有这样一个委托:

internal delegate void Feedback(int value);

编译器会生成一个类:

internal class Feedback : MultiCastDelegate {
public Feedback(object @object, IntPtr method);
// 用于同步回调
public virtual void Invoke(int value);
// 用于异步回调
public virtual IAsyncResult BeginInvoke(int value, AsyncCallback callback, object @object);
public virtual void EndInvoke(IAsyncResult result);
}

当创建一个委托时,编译器会把静态函数,成员函数等转化为这个类的实例对象。

注意这里的修饰符是internal和deletage的修饰符是一致的。

二、委托链

MultiCastDelegate

再来看下MultiCastDelegate这个基类,它有三个重要字段:

字段 类型 说明
_target object 如果引用一个实例方法,这里会存this
_methodPtr IntPtr 标示回调的方法
_invocationList object 用于表示委托数组

这个类又继承自Delegate,这是C#的历史遗留问题,有些地方还需要用到Delegate作为参数类型。

Delegate.Combine

使用void Delegate.Combine(Delegate a, Delegate b)方法可以创建一个委托链。如果有一个参数为null返回值为非null的那个参数;如果都不是null,返回值为一个新的委托,其_invocaionList指向一个数组,数组的元素为这两个参数。

a b 返回值
null delegate1 delegate1
delegate2 null delegate2
delegate1 delegate2 delegateChain

Delegate.Remove

通过Remove方法可以删除委托(链)中从后向前找到的第一个符合条件的委托

  1. 如果没有了,返回null
  2. 如果只剩一个委托,返回这个委托。
  3. 如果还有多余一个委托,创建一个新的委托,和对应的数组,把剩下的委托拷贝进去。

带返回值的委托

如果委托类型带有返回值,对一个委托链进行Invoke操作会返回委托链中最后一个委托的调用结果。

显式控制委托链的调用

Delegate[] MultiCastDelegate.GetInvocationList()

三、语法糖和lambda表达式

  1. 可以用+=-=表示Delegate.CombineDelegate.Remove方法。
  2. 隐式地使用静态函数或成员函数新建委托实例。
    void foo() {}
    void bar() {
    Action action = foo;
    action += foo;
    action -= foo;
    }

lambda表达式

int a = 0;
int b = 1;
Action action = ()=> { Console.WriteLine(a + b); };

分两种情况:

  1. 没有引用局部变量,lambda表达式会转化为类中的一个私有方法。

    1. 如果没有引用类的成员函数或成员变量,这是一个静态方法。
    2. 否则会成为一个成员方法。
  2. 如果lambda表达式引用了局部变量,编译器会为lambda表达式创建一个局部类。
    1. 类的成员变量用来存储引用的局部变量。
    2. 类的成员函数表示这个lambda表达式。

.NET via C#笔记17——委托的更多相关文章

  1. Ext.Net学习笔记17:Ext.Net GridPanel Selection

    Ext.Net学习笔记17:Ext.Net GridPanel Selection 接下来是Ext.Net的GridPanel的另外一个功能:选择. 我们在GridPanel最开始的用法中已经见识过如 ...

  2. SQL反模式学习笔记17 全文搜索

    目标:全文搜索 使用SQL搜索关键字,同时保证快速和精确,依旧是相当地困难. SQL的一个基本原理(以及SQL所继承的关系原理)就是一列中的单个数据是原子性的. 反模式:模式匹配 使用Like 或者正 ...

  3. JAVA自学笔记17

    JAVA自学笔记17 1.Map接口 1)概述 将键映射到值的对象,一个映射不能包含重复的键,每个键最多只能映射到一个值.可以存储键值对的元素 2)与Collection接口的不同: ①Map是双列的 ...

  4. golang学习笔记17 爬虫技术路线图,python,java,nodejs,go语言,scrapy主流框架介绍

    golang学习笔记17 爬虫技术路线图,python,java,nodejs,go语言,scrapy主流框架介绍 go语言爬虫框架:gocolly/colly,goquery,colly,chrom ...

  5. 【C#进阶系列】17 委托

    委托主要是为了实 现回调函数机制,可以理解为函数指针(唯一不同的在于多了委托链这个概念). 然而用的时候可以这么理解,但是委托的内部机制是比较复杂的. 一个委托的故事 delegate void ra ...

  6. [CLR via C#]17. 委托

        回调函数是一种非常有用的编程机制,它已经存在很多年了.Microsoft .NET Framework通过委托(delegate)来提供一种回调机制.不同于其他平台(比如非托管C++)的回调机 ...

  7. (.NET高级课程笔记)委托、事件总结

      1.委托的声明.实例化和调用 同样的,也可以把事务写成上面的形式 2.泛型委托---Func.Action 3.委托的意义:解耦 4.委托的意义:异步多线程 5.委托的意义:多播委托 6.观察者模 ...

  8. js-权威指南学习笔记17

    第十七章 事件处理 1.事件处理程序或事件监听程序是处理或响应事件的函数. 2.事件对象是与特定事件相关且包含有关该事件详细信息的对象. 3.响应通过键盘改变焦点的表单元素在得到和失去焦点时会分别出发 ...

  9. Linux下汇编语言学习笔记17 ---

    这是17年暑假学习Linux汇编语言的笔记记录,参考书目为清华大学出版社 Jeff Duntemann著 梁晓辉译<汇编语言基于Linux环境>的书,喜欢看原版书的同学可以看<Ass ...

随机推荐

  1. ubuntu16.04 使用tensorflow object detection训练自己的模型

    一.构建自己的数据集 1.格式必须为jpg.jpeg或png. 2.在models/research/object_detection文件夹下创建images文件夹,在images文件夹下创建trai ...

  2. [Educational Codeforces Round 81 (Rated for Div. 2)]E. Permutation Separation(线段树,思维,前缀和)

    [Educational Codeforces Round 81 (Rated for Div. 2)]E. Permutation Separation(线段树,思维,前缀和) E. Permuta ...

  3. 「NOIP2011」Mayan游戏

    传送门 Luogu 解题思路 爆搜,并考虑几个剪枝. 不交换颜色相同的方块(有争议,但是可以过联赛数据 \(Q \omega Q\)) 左边为空才往左换 右边不为空才往右换 因为对于两个相邻方块,右边 ...

  4. 096、Java中String类之手工入对象池操作

    01.代码如下: package TIANPAN; /** * 此处为文档注释 * * @author 田攀 微信382477247 */ public class TestDemo { public ...

  5. IOS UIKIT_EXTERN, __attribute__((visibility ("default"))) 是啥玩意?

    问题提出 在学习IOS时候,碰到一个函数NSStringFromCGPoint (UIGeometry.h) 其原型是 UIKIT_EXTERN NSString *NSStringFromCGPoi ...

  6. 七:日期类Date、日期格式化SimpleDateFormat、日历Calendar

    日期的格式转换:

  7. jmeter断言之响应code:200

    因为Jmeter只要检测到网页的响应,就认为是Pass而并不管当前网页内容的正确性.在进行压力测试时,为了检查Web服务器返回的网页是否正确,我们可以设置断言,这些断言是验证网页上是否存在指定的Tex ...

  8. LeetCode242 有效的字母异位词(Java字符数组排序&自定义排序记录)

    题目: 给定两个字符串 s 和 t ,编写一个函数来判断 t 是否是 s 的字母异位词.   示例 1: 输入: s = "anagram", t = "nagaram& ...

  9. windows制作动态链接库和使用一

    制作: //myDll.h _declspec(dllexport) int add(int a,int b); _declspec(dllexport) int sub(int a,int b); ...

  10. day07-Python运维开发基础(深/浅拷贝、字典/集合/相关操作)

    1. 深拷贝与浅拷贝 # ### 深拷贝 和 浅拷贝 """ a = 7 b = a a = 8 print(b) lst1 = [1,2,3] lst2 = lst1 ...