解析.Net 4.0 中委托delegate的使用详解

.Net中的委托从功能上讲和c语言或者c++中的方法指针类似,可以像调用方法一样调用委托完成某个功能,或返回某类结果。但是.Net毕竟是更高级的语言,委托Delegate也更高级了,委托是一种数据接口,它包含调用目标和调用方法的指针;而在.Net中定义的委托都继承自MulticastDelegate即多播委托,所谓的多播委托是指可以包含多个调用方法的委托。
一. 先来看下委托的定义:
如下C#代码定义委托
public delegate void DoSomething(int times);
委托的定义包含5个部分
1) public表示委托的可访问性
2) delegate关键字表示要定义一个委托
3) void表示委托定义方法的返回值
4) DoSomething是委托的名字
5) (int times) 是委托方法的参数列表,此处的参数列表可以包括ref参数,也可以有out参数,同样也可以有parms可变数量参数;需要注意如果委托中有多个调用方法,使用out参数时只能返回委托最后执行成功的一个委托方法的计算值
在C#中定义委托非常简单,只比方法定义的返回值之前多一个delegate关键字即可。
可是我们知道所有的用户定义委托都继承自MulticastDelegate;而MulticastDelegate是一个类;所以自定义的委托肯定也是一个类;看下上述代码的IL代码就可以证明我们的推断:


代码如下:

.class public auto ansi sealed delegates.DoSomething
    extends [mscorlib]System.MulticastDelegate
{
    // Methods
    .method public hidebysig specialname rtspecialname
        instance void .ctor (
            object 'object',
            native int 'method'
        ) runtime managed
    {
    } // end of method DoSomething::.ctor

.method public hidebysig newslot virtual
        instance void Invoke (
            int32 times
        ) runtime managed
    {

} // end of method DoSomething::Invoke

.method public hidebysig newslot virtual
        instance class [mscorlib]System.IAsyncResult BeginInvoke (
            int32 times,
            class [mscorlib]System.AsyncCallback callback,
            object 'object'
        ) runtime managed
    {

} // end of method DoSomething::BeginInvoke

.method public hidebysig newslot virtual
        instance void EndInvoke (
            class [mscorlib]System.IAsyncResult result
        ) runtime managed
    {

} // end of method DoSomething::EndInvoke

} // end of class delegates.DoSomething

二. 定义了委托,当然是为了使用它,来看下如何使用委托:
在.Net中有三种委托的形式,分别是方法、匿名方法和lambda表达式;我们用方法定义的形式看下委托的使用方法


代码如下:

using System;

namespace delegates
{
 public delegate void DoSomething(int times);

class Program
    {
        static void Main(string[] args)
        {
            //声明委托变量并给委托变量赋值
            DoSomething @do = DoA;
            //可以使用+号或者+=给委托增加方法
            @do += DoB;
            //执行委托时将按照委托的添加顺序先后执行委托中的方法
            @do(1);
            //也可以通过-号或者-= 从委托中移除方法
            @do -= DoA;
            @do(2);

@do -= DoB;
            //将委托中的所有方法都移除掉之后,委托照样是可以调用的,只是什么都不做
            @do(3);

Console.Read();
        }
       //定义一个委托相同参数和返回值的方法
        static void DoA(int times)
        {
            Console.WriteLine("Do A {0}", times);
        }

//定义一个委托相同参数和返回值的方法
        static void DoB(int times)
        {
            Console.WriteLine("Do B {0}", times);
        }
    }
}

如上代码中的Main方法,首先我们定义了委托DoSomething的变量@do,并将DoA方法直接赋值给此委托变量;然后我们又使用+=符号或者+号给此委托添加了另一个方法;当然也可以使用-或者-=从委托中去掉方法。
委托比C/C++方法指针强大的地方在于其可以容纳多个方法,也可以执行+/-操作从方法列表中添加或者删除掉方法。
在执行委托加减运算时有几个问题需要我们注意:
1. 委托声明的写法
委托声明时可以用如下写法


代码如下:

DoSomething @do = DoA;

这其实是一种简短的写法,我们知道在.Net 1.x中这样写是不允许的只有到.Net 2.0时才允许这么写,在.Net 1.x中必须写成


代码如下:

DoSomething @do = new DoSomething(DoA);

我们要在声明时就给@do赋予DoA加上DoB


代码如下:

DoSomething @do = DoA + DoB;

这么写是不行的,编译器不干了;必须使用.Net 1.x中的写法


代码如下:

DoSomething @do = new DoSomething(DoA) + new DoSomething(DoB);

2. 从委托中减去委托中本不存在的方式时会发生什么呢?
请看如下代码:


代码如下:

DoSomething @do = DoA;
@do -= DoB;

第一行代码我生命了@do并将DoA赋予它;第二行代码我尝试从@do中减去DoB,DoB并没有在@do的方法列表中存在,这样会发生什么情况呢?首先编译器没有报错,程序可以正常的编译;执行代码发现可以程序可以正常执行,调用@do委托时正确的执行了DoA方法;这说明了.Net包容了我们程序员犯的错,我们从委托变量中减去一个委托中并不包含的方法时,不会报错会正常的执行。
3. 对委托做减法,所有委托都减完了,会怎样呢?看如下代码


代码如下:

DoSomething @do = new DoSomething(DoA) + new DoSomething(DoB);
@do -= DoA;
@do -= DoB;
@do(1);

这样的代码可以成功编译,但是在运行时会报NullReferenceException;这显然不是我们希望的,所以对委托做减法时要特别注意。


代码如下:

<span style="text-decoration: line-through;">public delegate void DoIt(string task);

class Test
{
    static void Main(string[] args)
    {
        //DoIt声明,赋予一个参数更宽泛的方法是合法的
        DoIt doIt = new DoIt(DoItImpl);
        doIt("hello");
    }

//比委托定义中的参数更宽泛,string类型可以隐式转换成object
    static void DoItImpl(object task)
    {
        Console.WriteLine("DoItImpl {0}",task);
    }
}
</span>

(0)

相关推荐

  • 深入c# Func委托的详解

    如下所示: 复制代码 代码如下: using System;using System.Collections.Generic;using System.Linq;using System.Text;namespace Anonymous_Lam{    delegate string ConvertMethod(string Method);    class Lambda_Fun    {        static void Main()        {            Conver

  • 对c#中委托的理解

    理解委托从一个简单的例子开始 金城武演的有部老电影叫<薰衣草>,里面有个情节大概是这样的:小金收客户的钱,然后代表客户去向不同的人Say I love you. 一开始他的客户都是中国人,只需要说中文,如下代码示例,很简单,支持所有中国客户: 复制代码 代码如下: public class LoveManager{    public void Love(string name)    {        Console.WriteLine("我爱你, {0}", name)

  • C#委托初级使用的实例代码

    复制代码 代码如下: delegate double ProcessDelegate(double param1, double param2); static double Muliply(double param1, double param2)        {            return param1 * param2;                } static double Divide(double param1, double param2)        {    

  • 基于C#委托的深入分析

    1.委托的定义 委托可以看成是一种数据类型,可以用于定义变量能接受的值只能是一个方法. 委托简单的示例: 复制代码 代码如下: namespace DelegateDemo {     class Program     {         public delegate int MathOptDelegate(int value1,int value2);         public int add(int value1, int value2)         {            

  • 深入理解C#中常见的委托

    一提到委托,浮现在我们脑海中的大概是听的最多的就是类似C++的函数指针吧,呵呵,至少我的第一个反应是这样的.关于委托的定义和使用,已经有诸多的人讲解过,并且讲解细致入微.我就不用多废话了.今天我要说的是C#中的三种委托方式:Func委托,Action委托,Predicate委托以及这三种委托的常见使用场景.Func,Action,Predicate全面解析首先来说明Func委托,通过MSDN我们可以了解到,Func委托有如下的5种类型: 复制代码 代码如下: (1) *delegate TRes

  • C#零基础学习理解委托

    说来惭愧,在大学的课程中,竟然没有听说过委托这个名称.那么今天我就带着大家一起探讨下委托和事件. 咱们先来看下委托 我主要从以下几个方面讲解 1,  为什么使用委托  2.什么是委托  3.委托如何使用 为什么使用委托? 委托是c#中非常重要的一个概念,使用委托使程序员可以将方法引用封装在委托对象内.然后可以将该委托对象传递给可调用所引用方法的代码,而不必在编译时知道将调用哪个方法.与C或C++中的函数指针不同,委托是面向对象,而且是类型安全的. 什么是委托? 委托是一种引用方法的类型,一旦为委

  • 经典的委托排序(深入分析)

    对于数值型的排序我们都已经司空见惯了,但我们有时候希望我们的Sort()方法能够给任何对象排序,比如某段客户机代码包含Currency结构数组或其他的类和结构,就需要对该数组排序.这里我们使用委托并封装这个方法进行比较. 排序我们还是使用经典的冒泡排序,如果数据量较大你可以换为你自己的更高效的排序算法. 先给出整个代码: 复制代码 代码如下: public class BubbleSorter{ public static void Sort(object[] sortArray, Compar

  • 深入委托与多播委托的详解

    复制代码 代码如下: using System;using System.Collections.Generic;using System.Text;namespace ConsoleApplication1{    class 简单的委托实例    {        public static double Area(double Val)        {            return Val * 2;        }        public static double Scar

  • 解析.Net 4.0 中委托delegate的使用详解

    .Net中的委托从功能上讲和c语言或者c++中的方法指针类似,可以像调用方法一样调用委托完成某个功能,或返回某类结果.但是.Net毕竟是更高级的语言,委托Delegate也更高级了,委托是一种数据接口,它包含调用目标和调用方法的指针:而在.Net中定义的委托都继承自MulticastDelegate即多播委托,所谓的多播委托是指可以包含多个调用方法的委托.一. 先来看下委托的定义:如下C#代码定义委托public delegate void DoSomething(int times);委托的定

  • vue2.0 中#$emit,$on的使用详解

    vue1.0中 vm.$dispatch 和 vm.$broadcast 被弃用,改用$emit,$on vm.$on( event, callback ) 监听当前实例上的自定义事件.事件可以由vm.$emit触发.回调函数会接收所有传入事件触发函数的额外参数. vm.$emit( event, [-args] ) 触发当前实例上的事件.附加参数都会传给监听器回调. 例子: //父组件 <template> <ratingselect @select-type="onSele

  • jQuery3.0中的buildFragment私有函数详解

    时隔 3 个月,jQuery 团队终于发布了 3.0 Alpha 版本.有两个版本 jQuery compat 3.0 和 jQuery 3.0. jQuery compat 3.0 对应之前的 1.x, 兼容更多的浏览器,对于IE支持到 8.0 版本 jQuery 3.0 对应之前的 2.x,关注更新的浏览器,对于IE支持到 9.0 版本 此外, 3.0还增加了对 Yandex 浏览器的支持,一款来自俄罗斯的浏览器. 下面看下jQuery3.0中的buildFragment. 在 jQuery

  • pytorch1.0中torch.nn.Conv2d用法详解

    Conv2d的简单使用 torch 包 nn 中 Conv2d 的用法与 tensorflow 中类似,但不完全一样. 在 torch 中,Conv2d 有几个基本的参数,分别是 in_channels 输入图像的深度 out_channels 输出图像的深度 kernel_size 卷积核大小,正方形卷积只为单个数字 stride 卷积步长,默认为1 padding 卷积是否造成尺寸丢失,1为不丢失 与tensorflow不一样的是,pytorch中的使用更加清晰化,我们可以使用这种方法定义输

  • vue3.0中友好使用antdv示例详解

    前言 随着我们vue3.0的出现,我们的ui组件库也有了一些变化,像我们的旧版的element-ui已经不能在vue3.0中使用了,如果要使用element的话需要使用最新版的element-plus,由于发现它并不太好用,因此我选择了Ant Design Vue. 如果我们以前经常使用antd的话,我们使用起来这个上手会非常方便. 在vue3.0中引入我们的antdv 1.首先使用我们的vue/cli创建vue3.0项目并使用less 2. 在vue3.0中使用的话我们需要安装 ant-des

  • C语言中指针 int *p=0;和int *p;*p=0;和”&“的关系和区别详解

    初学者在学习C语言的时候,最头疼的可能就是指针,话不多说.让我们直接进入正题 直接上代码 int main(void) { int *p = 0; printf("%d", *p); system("pause"); return 0; } 直接运行,好了,程序是不是报错了?那就对了.因为此时的int *p=0实际等于int *p; p=0; 让我们来验证一下对不对 int main(void) { int *p = 0; printf("%d"

  • SpringBoot中对应2.0.x版本的Redis配置详解

    properties格式: # REDIS (RedisProperties) # Redis数据库索引(默认为0) spring.redis.database=0 # Redis服务器地址 spring.redis.host=localhost # Redis服务器连接端口 spring.redis.port=6379 # Redis服务器连接密码(默认为空) spring.redis.password= # 连接池最大连接数(使用负值表示没有限制) spring.redis.jedis.po

  • QT委托代理机制之Model View Delegate使用方法详解

    目录 本地数据加载(Data) 添加数据模型(Model) 添加代理模型(Proxy) 添加元素的代理(Delegate) 添加视图层(View) 使用效果 之前的一篇文章中介绍过QT的委托代理机制,那时候由于理解的比较浅就简单的给了一个例子.最近又做了一部分相关的工作,发现之前的理解有点问题.这里就详细的介绍一下QT的委托代理机制的用法,希望对大家有帮助. Model-View-Delegate机制可以简单的理解为将本地的一些数据以特定的UI形式呈现出来.常见的数据结构包括列表数据(list)

  • Oracle中游标Cursor基本用法详解

    查询 SELECT语句用于从数据库中查询数据,当在PL/SQL中使用SELECT语句时,要与INTO子句一起使用,查询的 返回值被赋予INTO子句中的变量,变量的声明是在DELCARE中.SELECT INTO语法如下: SELECT [DISTICT|ALL]{*|column[,column,...]} INTO (variable[,variable,...] |record) FROM {table|(sub-query)}[alias] WHERE............ PL/SQL

  • Android通过json向MySQL中读写数据的方法详解【读取篇】

    本文实例讲述了Android通过json向MySQL中读取数据的方法.分享给大家供大家参考,具体如下: 首先 要定义几个解析json的方法parseJsonMulti,代码如下: private void parseJsonMulti(String strResult) { try { Log.v("strResult11","strResult11="+strResult); int index=strResult.indexOf("[");

随机推荐