委托

最近自己在调试C#项目,发现经常可以看到委托和lambda表达式,各种花里胡哨的写法把我给整的云里雾里的,于是自己特意花了一点功夫来整理关于delegate的相关知识,方便自己日后查阅。

何为委托

委托是.NET中的寻址方法,和C++的函数指针很像;但是委托是类型安全的类,定义了返回类型和参数类型,也就是说委托一种用户自定义的类型,和普通的类一样;

委托的使用

声明委托

    delegate void IntMethodInvoker(int x);    //定义委托IntMethodInvoker, 参数是int,返回类型void

我们可以把委托当做是一件事情,就是给方法名字和返回类型指定一个别名

从上面的例子可以总结出声明委托的模板如下:

  • delegate + 返回类型 + 委托类型名 + 参数列表

委托的使用

实例化委托,并且需要对其进行初始化,这样可以类似地当做一个变量使用了,下面给出一个具体的例子来说明如何使用委托。

    using System;

    namespace DelegateSamples{

        delegate double DoubleOp(double x);    // 声明一个委托类型
class MathOperations
{
public static doube MultiplyByTwo(double value)
return value * 2; public static double Square(double value)
return value * value;
} class Program
{
static void Main()
{
// 声明一个委托数组,就像普通数组一样
DoubleOp[] operations = {MathOperations.MultiplyByTwo, MathOperations.Square}; for(int i = 0; i < operations.length; i++)
{
Console.WriteLine("using operations[{0}]:", i);
ProcessAndDisplayNumber(operations[i], 3.2);
Console.WriteLine();
}
} static void ProcessAndDisplayNumber(DoubleOp action, double value)
{
double res = action(value); // 调用action实际封装的方法
Console.WrireLine("Value is {0}, result of operation is {1}", value, res);
}
}

在调用委托的时候,最后判断委托是否为null,不然可能会引起异常

Action和Func委托

Action<T>委托表示引用一个void返回类型的方法,Action<T>有很多种变体,Action<in T>调用带一个参数的方法,没有泛型参数的Action类调用不带参数的方法;

Func<T>委托类似,表示引用带返回类型的方法,Func<T>也有很多变体,Func<Out TResult>表示调用带返回类型但没有参数的方法,Func<in T, out TRsult>调用一个带参数的方法,以此类推;

// 声明一个返回double类型,并且带一个double参数的委托
Func <double, double> operations = {MathOperations.MultiplyByTwo, MathOperations.Square}; // 注意这个方法和上面方法的不一样,利用了Func类
static void ProcessAndDisplayNumber(Func <double, double> action, double value)
{
double res = action(value); // 调用action实际封装的方法
Console.WrireLine("Value is {0}, result of operation is {1}", value, res);
}

多播委托

在之前的例子中,每个委托都只含有一个方法调用,也就是说调用多少次委托就是调用多少次方法。

但是,一个委托也可以包含多个方法,这种委托叫做多播委托,因此,这种委托必须返回void,不然返回的数据是不对的

    Action<double> operations = MathOperations.MultiplyByTwo;
operations += MathOperations.Square; //这里在给委托添加一个方法

对于多播委托的使用,调用方法链的顺序并没得到正式定义,这要求我们尽量避免依赖特定顺序调用方法的代码;

从上面的例子可以知道,多播委托是可以识别+,+=,-,-=这些符号的,下面给出具体的示意:

  • +=: 表示委托新增一个方法
  • -=: 表示委托减少一个方法

匿名方法

在上面的例子中,委托所调用的方法,都是我们自己预先写好的,但是委托也可以使用匿名方法,也就是将匿名方法用作委托的参数,这在实例化委托的会有一些不一样的,具体如下面例子所示:

    string name = "";

    // Func<string, string>委托接受一个string参数,返回一个string,注意匿名方法的实现,delegate开头
Func<string, string> anonDel = delegate(string param){
param += name;
param += "just for test";
return param;
};

使用匿名方法需要遵守的规则:

  1. 在匿名方法中不能使用跳转语句(goto、break、continue),匿名方法外部的语句也不能跳转到匿名方法内部;
  2. 在匿名方法中不能访问不完全的代码,也不能访问在匿名方法外部定义的ref、out参数;

在C#3.0之后,lambda表达式代替了匿名方法,写起来感觉更舒服;

Lambda表达式

lambda表达式主要是用来替代匿名方法的,因为显然委托知道他自己需要调用的是什么方法,不需要声明delegate关键字,在参数和方法体之间插入=>,表示“goes to",具体示例如下所示:

    string name = "";

    // Func<string, string>委托接受一个string参数,返回一个string,这里使用lambda表达式
Func<string, string> anonDel = param => {
param += name;
param += "just for test";
return param;
};
// 这样的lambda表达式是不是很优雅,就是调试程序的时候会有点烦

为了更好地使用lambda表达式,方便我们自己写代码写的花里花哨的,就对其多做一点介绍:

参数

  1. 如果只有一个参数,就只需要写出参数名字就行

     Func<string, string> oneParam = s => String.Format("change to Upper {0}", s.ToUpper());
  2. 使用多个参数,就把参数用写在()里

    Func<double, double, double> twoParams= (x, y) => x * y;    // 返回x*y

在使用多个参数的时候,我觉得还是在参数前面加上类型比较好,便于理解,例如(double x, double y) => x * y;

多行代码

  1. 只有一条语句,方法里面不需要{}和return,编译器会自动添加一个隐式的return

     Func<double, double, double> result = x, y => x * y;
  2. 含有多条语句,需要加上{}和return

    string lastname = "Alex";
    Func<string ,string> printName = name =>
    {
    name += lastname;
    return String.Format("Being Upper: {0}",name);
    }

c#学习笔记之委托的更多相关文章

  1. 《C# 语言学习笔记》——委托

    委托是一种可以把引用存储为函数的类型. 委托的声明非常类似于函数,但不带函数体,且要使用delegate关键字.委托的声明制定了一个返回类型和一个参数列表. 在定义了委托后,就可以声明该委托类型的变量 ...

  2. [读书笔记]C#学习笔记二: 委托和事件的用法及不同.

    前言:  C#委托是什么 c#中的委托可以理解为函数的一个包装, 它使得C#中的函数可以作为参数来被传递, 这在作用上相当于C++中的函数指针. C++用函数指针获取函数的入口地址, 然后通过这个指针 ...

  3. [C#学习笔记]Func委托与Action委托

    学习一项新知识的时候,最好的方法就是去实践它. 前言 <CLR via C#>这本神书真的是太有意思了!好的我的前言就是这个. Fun 如果要用有输入参数,有返回值的委托,那么Func委托 ...

  4. js学习笔记-事件委托

    通过事件委托,你可以把事件处理器绑定到父元素上,避免了把事件处理器添加到多个子级元素上.从而优化性能. 事件代理用到了事件冒泡和目标元素.而任何一个元素的目标元素都是一开始的那个元素. 这里首先要注意 ...

  5. 【c# 学习笔记】委托链的使用

    委托链其实就是委托类型,只是委托链把多个委托链接在一起而已,也就是说,我们把链接了多个方法的委托称为委托链或多路广播委托.如下: public delegate void DelegateTest() ...

  6. 【c# 学习笔记】委托的使用

    //委托使用的演示 class Program { //1.使用delegate关键字来定义一个委托类型 public delegate void MyDelegate(int para1, int ...

  7. c#学习笔记03——委托和事件

    委托:一种引用类型,这种类型可以用来定义方法签名,从而使用委托实现将方法作为参数传递给其他方法.类似于C++中的函数之争,使用委托使程序员可以将方法引用封装在委托对象内. 定义和声明委托: deleg ...

  8. 委托学习笔记后续:泛型委托及委托中所涉及到匿名方法、Lambda表达式

    引言: 最初学习c#时,感觉委托.事件这块很难,其中在学习的过程中还写了一篇学习笔记:委托.事件学习笔记.今天重新温故委托.事件,并且把最近学习到和委托相关的匿名方法.Lambda表达式及泛型委托记录 ...

  9. C#委托与事件学习笔记

    委托事件学习笔记 本文是学习委托和事件的笔记,水平有限,如有错漏之处,还望大神不吝赐教. 什么是委托?从字面意思来解释,就是把一个动作交给别人去执行.在实际开发中最常用的就是使一个方法可以当做一个参数 ...

随机推荐

  1. Android通过子线程更新UI的几种方式

    一般情况下,UI的更新都少不了Handler,首先我们先了解一下Handler机制: Handler消息机制 定义 Message 线程间通信的数据单元,可通过message携带需要的数据创建对象:M ...

  2. 2019 ICCV、CVPR、ICLR之视频预测读书笔记

    2019 ICCV.CVPR.ICLR之视频预测读书笔记 作者 | 文永亮 学校 | 哈尔滨工业大学(深圳) 研究方向 | 视频预测.时空序列预测 ICCV 2019 CVP github地址:htt ...

  3. 一文熟练使用python mock

    mock作为python测试模拟对象工具,在单元测试当中使用较多,官方文档详细不够精简,这篇文章介绍mock常用的用法,以下为引用全文,留给自己和有需要的人查阅. https://realpython ...

  4. JS 头像显示

    HTML <div class="form-group"> <label class="col-sm-3 control-label"> ...

  5. lisp学习总结(二)-----lisp应该探索发展的方向

    现在流行一种语言叫做Clojure,他是lisp直接嫁接到java的结果,但是我就感觉这却成为lisp的失败. 因为lisp最强大最有优势的能力是构造抽象,构造设计思想,而不是运行期以确定的方式运行, ...

  6. 【Java并发基础】死锁

    前言 我们使用加锁机制来保证线程安全,但是如果过度地使用加锁,则可能会导致死锁.下面将介绍关于死锁的相关知识以及我们在编写程序时如何预防死锁. 什么是死锁 学习操作系统时,给出死锁的定义为两个或两个以 ...

  7. 人生中的第一篇OI博客及博客规划

    这是笔者第一次在博客园里发表文章,也同样是第一次来写关于OI的一些想法,此篇的主题是想总体对日后的博客有具体的规划. 首先,笔者创办博客并发表观点于看法的目的是记录自己对于题目或竞赛的观念,主要以题解 ...

  8. Python3基础之内置模块

    模块和包 一.定义: 模块:用来从逻辑上组织Python代码(变量,函数,类,逻辑:实现一个功能),本质就是.py结尾的Python文件包:用来从逻辑上组织模块,本质就是一个目录(必须带有一个__in ...

  9. php--->自己封装的简易版mvc框架

    最近根据自己的理解,封装了一个自己的框架,来重新系统化梳理自己对mvc框架的理解:后续会陆续添加各种新的功能. 欢迎指点交流. GitHub:https://github.com/Frankltf/m ...

  10. vscode打开文件,中文显示乱码(已解决)

    之前使用vscode打开keil的文件后,发现显示乱码,网上查找资料发现大多是这种方法:将files.autoGuessEncoding改为true,但是并没有用. 发现第二种方法为:在vscode中 ...