对c#中委托的理解

理解委托从一个简单的例子开始

金城武演的有部老电影叫《薰衣草》,里面有个情节大概是这样的:小金收客户的钱,然后代表客户去向不同的人Say I love you。

一开始他的客户都是中国人,只需要说中文,如下代码示例,很简单,支持所有中国客户:

代码如下:

public class LoveManager
{
    public void Love(string name)
    {
        Console.WriteLine("我爱你, {0}", name);
    }
}

代码如下:

class Program
{
    static void Main(string[] args)
    {
        LoveManager loveManager = new LoveManager();
        loveManager.Love("张曼玉");
    }
}

执行结果:

代码如下:

我爱你, 张曼玉

我留意到后来电影里出现了外国客户,我想代码应该是这样:

代码如下:

//枚举,可扩展多语种
public enum Language
{
    English,
    Chinese
}

代码如下:

public class LoveManager
{
    public void Love(string name, Language lang)
    {
        switch (lang)
        {
            case Language.Chinese:
                loveChinese(name);
                break;
            case Language.English:
                loveEnglish(name);
                break;
        }
    }

//汉语客户专用
    public void LoveChinese(string name)
    {
        Console.WriteLine("我爱你, {0}", name);
    }

//英语客户专用
    public void LoveEnglish(string name)
    {
        Console.WriteLine("I love you, {0}", name);
    }
}

代码如下:

class Program
{
    static void Main(string[] args)
    {
        LoveManager loveManager = new LoveManager();
        loveManager.Love("张曼玉", Language.Chinese);
        loveManager.Love("Sophie Marceau", Language.English);
    }
}

执行结果:

代码如下:

我爱你, 张曼玉
I love you, Sophie Marceau

OK,现在张曼玉能听懂“我爱你”,Sophie Marceau能听懂“I love you”。虽然支持了英汉双语表白,但以后还有法国客户,葡萄牙客户,阿拉伯客户怎么办?每扩展一个语种除了添加这个语种“我爱你”的方法,还得扩展枚举,扩展LoveManager.Love(),确实有些繁琐。

C语言时代:指针

此时,不得不提到C语言中大名鼎鼎的指针。指针允许把一个函数的地址作为参数传递给另一个函数,这个特性在以后的各种高级语言中得到了扩展和加强。先看如下C代码:

代码如下:

#include <stdio.h>

//接受一个指针类型的参数
void func1(void(*p)(void)){
    printf("this is func1\r\n");
    //通过指针调用函数
    p();
}

void func2(){
    printf("this is func2\r\n");
}

int main() {
    //将func2地址作为参数传递
    func1(func2);
    return 0;
}

执行结果:

代码如下:

this is func1
this is func2

在.Net中能不能像C语言一样,把函数作为一个参数传递并且调用呢?

代码如下:

//这段代码并不能被执行,但如果在.Net中可以这样写的话问题就会简单很多 Love("张曼玉", LoveChinese);
Love("Sophie Marceau", LoveEnglish);

.Net中更完美的解决方案:委托

在.Net中不但可以像C语言一样将函数作为参数传递,并且.Net提供了类型安全机制和更加强大的功能,如下提供了使用委托的完整代码示例:

代码如下:

using System;

namespace DelegateDemo
{
    //定义委托
    public delegate void LoveDelegate(string name);

public class LoveManager
    {
        public void Love(string name, LoveDelegate loveDelegate)
        {
            loveDelegate(name);
        }

//汉语客户专用
        public void LoveChinese(string name)
        {
            Console.WriteLine("我爱你, {0}", name);
        }

//英语客户专用
        public void LoveEnglish(string name)
        {
            Console.WriteLine("I love you, {0}", name);
        }
    }

class Program
    {
        static void Main(string[] args)
        {
            LoveManager loveManager = new LoveManager();
            loveManager.Love("张曼玉", loveManager.LoveChinese);
            loveManager.Love("Sophie Marceau", loveManager.LoveEnglish);
        }
    }
}

执行结果:

代码如下:

我爱你, 张曼玉
I love you, Sophie Marceau

定义委托

代码如下:

public delegate void LoveDelegate(string name);

我们现在对委托做一个总结:
委托是一个类,它定义了方法的类型,使得可以将方法当作另一个方法的参数来进行传递,这种将方法动态地赋给参数的做法,可以避免在程序中大量使用If-Else(Switch)语句,同时使得程序具有更好的可扩展性。

在C#中委托使用特有的关键字 delegate 来定义,在delegate之后紧跟的是函数签名。为了确保类型安全,.Net中的委托要求函数具有相同的签名,比如 func(int p) 和func(string p)不能使用同一个委托,因为它们的参数类型不一样。

通过ILDasm.exe可以发现,定义委托的那行代码实际在编译时会自动生成一个类,如果要还原这个类,代码会是这样:

代码如下:

public class LoveDelegate : System.MulticastDelegate
{
      //构造器
      public LoveDelegate(Object obj, IntPtr method);

//原型
      public virtual void Invoke(string name);

//异步回调
      public virtual IAsyncResult BeginInvoke(Int32 value, AsyncCallback callback, Object obj);
      public virtual void EndInvoke(IAsyncResult result);
}

因此,委托实际上就是一个类,它继承至System.MulticastDelegate,凡是可以定义类的地方,都可以定义委托。

委托的构造函数

代码如下:

LoveManager loveManager = new LoveManager();
//编译不能通过,委托必须使用带有一个参数的构造函数
//LoveDelegate loveDelegate = new LoveDelegate();
LoveDelegate loveDelegate = new LoveDelegate(loveManager.LoveChinese);
loveDelegate("吴剑");

与类不同的是,委托必须使用带有一个参数的构造函数。

委托推断语法

代码如下:

LoveManager loveManager = new LoveManager();
//等同于:LoveDelegate loveDelegate = new LoveDelegate(loveManager.LoveChinese);
LoveDelegate loveDelegate = loveManager.LoveChinese;
loveDelegate("吴剑");

委托与方法进行绑定

回到上面的例子,有一天一富二代找到小金,说钱不是问题,你去张曼玉楼下,用中文喊一遍,再用英文喊一遍。

代码如下:

static void Main(string[] args)
{
        LoveManager loveManager = new LoveManager();
        //定义委托变量
        LoveDelegate delegate1;
        //变量初始化(用中文喊一遍)
        delegate1 = loveManager.LoveChinese;
        //绑定方法(用英文再喊一遍)
        delegate1 += loveManager.LoveEnglish;
        delegate1("张曼玉");
}

执行结果:

代码如下:

我爱你, 张曼玉
I love you, 张曼玉

我们可以用 += 将多个方法绑定到一个委托,也可以使用  -= 移除方法与委托的绑定。

匿名方法

客户的需求总是千变万化,一个客户跟小金说,我要跟曼玉表白,除了用中英文,能不能后面再给我加一句,曼玉一听到这句准会答应我。

代码如下:

LoveManager loveManager = new LoveManager();
LoveDelegate loveDelegate = loveManager.LoveEnglish;
loveDelegate += loveManager.LoveChinese;
loveDelegate += delegate(string name)
{
    Console.WriteLine("{0}, 还记得大明湖畔的夏雨荷吗?", name);
};
loveDelegate("曼玉");

执行结果:

代码如下:

I love you, 曼玉
我爱你,曼玉
曼玉,还记得大明湖畔的夏雨荷吗?

针对这位特殊客户使用了匿名方法,不是每个人示爱的时候都会提到大明湖畔的夏雨荷,也就是这位特殊客户使用一次而以,所以没有必要定义一个独立的方法。使用匿名方法可以减少编码量,降低代码复杂度。

Lambda(λ)表达式

C# 3.0为匿名方法提供了Lambda表达式,如下代码执行结果与上面的示例完全一致:

代码如下:

LoveManager loveManager = new LoveManager();
LoveDelegate loveDelegate = loveManager.LoveEnglish;
loveDelegate += loveManager.LoveChinese;
//用红色字体标出了Lambda表达式部分loveDelegate += name =>
{
    Console.WriteLine("{0}, 还记得大明湖畔的夏雨荷吗?", name);
};
loveDelegate("曼玉");

=>为Lambda运算符,运算符左边列出匿名方法需要的参数,可以这样使用:

(string param1, int param2)

也可以:

(param1, param2)

如示例代码只有一个参数还可以去掉括号:

param1

Lambda表达式右边为匿名方法实现代码,如果实现代码只有一行,还可以删除花括号和return语句,因为编译器会自动添加。

共同学习,共同进步!

(0)

相关推荐

  • C# 委托的三种调用示例(同步调用 异步调用 异步回调)

    首先,通过代码定义一个委托和下面三个示例将要调用的方法: 复制代码 代码如下: public delegate int AddHandler(int a,int b);    public class 加法类    {        public static int Add(int a, int b)        {            Console.WriteLine("开始计算:" + a + "+" + b);            Thread.Sl

  • C#中委托的基本用法总结

    原则: 1.委托本质就是个指针,一个函数指针,拿到函数的首地址即可: C#的委托加了安全性,体现在对于函数指针所引用的函数指令块的类型检测,比如返回值,参数类型,参数个数 而C中的函数指针被赋值的时候(在C#中,就是委托实例化的时候,因为C#中后台将委托处理成一个类了,封装了哈)被赋予的值是否满足类型的种种条件(返回值,参数类型,参数个数)不做检查,由用户给出保证,C#会编译提示出来 2.委托实例化的过程,也就是委托对象构造的过程,从底层来讲,即将一个现存的函数代码指令块的内存地址(静态函数,实

  • c#委托学习示例分享

    1.委托 总的来说,委托是一个类,它定义了方法的类型,使得可以将方法当作另一个方法的参数来进行传递,这种将方法动态地赋给参数的做法,可以避免在程序中大量使用If-Else(Switch)语句,同时使得程序具有更好的可扩展性.所以,引入委托后,编程人员可以把方法的引用封装在委托对象中,然后把委托对象传递给需要引用方法.调用委托和调用方法的方式是一模一样的,代码如下: a.代码: 复制代码 代码如下: using System;using System.Collections.Generic;usi

  • c#委托详解和和示例分享

    什么是委托? 委托是寻址方法的.NET版本,使用委托可以将方法作为参数进行传递.委托是一种特殊类型的对象,其特殊之处在于委托中包含的只是一个活多个方法的地址,而不是数据. 委托虽然看起来像是一种类型,但其实定义一个委托,是定义了一个新的类.下面这行代码,定义了一个委托,使用ILDasm.exe查看其生成的IL代码如图所示: 复制代码 代码如下: //定义委托,它定义了可以代表的方法的类型,但其本身却是一个类 public delegate int methodDelegate(string st

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

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

  • C#中常见的系统内置委托用法详解

    一般来说,C#在公共语言运行时(CLR)环境中系统为我们内置了一些常用的委托,包括Action类的委托.Func类的委托.Predicate<T>委托.Comparison<T>委托等等.以上这些委托的命名空间都是System,所属程序集都是 mscorlib.dll,今天本文就来讲一讲这些委托的使用方法. 就像我们自己已定义好的一样,要实现某些功能,我们可以直接利用系统内置委托,实例化它们,而不必显式定义一个新委托并将命名方法分配给该委托.如: public static voi

  • C#委托delegate实例解析

    所谓c#的委托就是说把函数当参数来传递. 这个在js完全就用不着搞什么委托东西,直接转就是了.而对于C#来说则不是这样! 一个函数,如果它的参数是函数,那么是这样子写的 : public void method(Action<string, Int32> voidMethod, Func<string, Int32> returnMethod) Action<string, Int32> voidMethod 意思是说这个将被传进来的函数是一个没有return的函数,就

  • C#基础之委托用法实例教程

    本文以实例形式简单介绍了C#中委托的用法,是深入学习C#程序设计所必须掌握的重要技巧.现以教程形式分享给大家供大家参考之用.具体如下: 首先,委托是C#中最为常见的内容.与类.枚举.结构.接口一样,委托也是一种类型.类是对象的抽象,而委托则可以看成是函数的抽象.一个委托代表了具有相同参数列表和返回值的所有函数.比如: delegate int GetCalculatedValueDelegate(int x, int y); 在上面的定义中,我们定义了一个委托,这个委托代表着一类函数,这些函数的

  • C#中委托和事件在观察者模式中的应用实例

    通常来说当一个被监视对象的方法执行会触发观察者Observer的方法的时候,我们就可以在被监视对象中声明委托和事件.本文就以实例形式展示了C#中实现委托和事件在观察者模式中的应用.具体如下: 示例如下: 有一个宠物追踪器挂宠物身上,只要宠物离开主人100米之外,主人手上的显示器显示警告信息并声音报警. class Program { static void Main(string[] args) { PetTracker tracker = new PetTracker(); tracker.I

  • c#并行任务多种优化方案分享(异步委托)

    遇到一个多线程任务优化的问题,现在解决了,分享如下. 假设有四个任务: 任务1:登陆验证(CheckUser) 任务2:验证成功后从Web服务获取数据(GetDataFromWeb) 任务3:验证成功后从数据库获取数据(GetDatFromDb) 任务4:使用2.3的数据执行一个方法 (StartProcess) 一个比较笨的方法(本人最开始的方法,记为方法1)是直接开启一个线程,按照顺序依次执行四个任务: 复制代码 代码如下: new Thread(delegate              

  • c#委托把方法当成参数(实例讲解)

    静态方法代理: 复制代码 代码如下: public delegate void DoGreeting(string name); class Program    {        [STAThread]        static void Main(string[] args)        {            //方法名当成参数传给委托类型调用            MarkGreeting("张三", GreetingEnglish);            MarkGr

随机推荐