对c#中托的理解

 更新时间:2013年04月12日 15:46:23   投稿:jingxian  
本篇文章,小编将为大家介绍关于对c#中托的理解,有需要的朋友可以参考一下

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

复制代码 代码如下:

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语句,因为编译器会自动添加。

共同学习,共同进步!

相关文章

  • C#中BitmapImage与BitmapSource接口的区别对比小结

    C#中BitmapImage与BitmapSource接口的区别对比小结

    BitmapImage和BitmapSource都可以用于表示和显示图像,本文就来介绍一下C#中BitmapImage与BitmapSource接口的区别对比,具有一定的参考价值,感兴趣的可以了解一下
    2024-03-03
  • C#使用CallContext缓存线程数据

    C#使用CallContext缓存线程数据

    这篇文章介绍了C#使用CallContext缓存线程数据的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-05-05
  • C# 如何设置label(标签)控件的背景颜色为透明

    C# 如何设置label(标签)控件的背景颜色为透明

    这篇文章主要介绍了C# 如何设置label(标签)控件的背景颜色为透明,帮助大家更好的理解和使用c#,感兴趣的朋友可以了解下
    2020-10-10
  • c#中LINQ的基本用法(三)

    c#中LINQ的基本用法(三)

    这篇文章介绍了c#中LINQ的基本用法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下的相关资料
    2022-04-04
  • C#实现char字符数组与字符串相互转换的方法

    C#实现char字符数组与字符串相互转换的方法

    这篇文章主要介绍了C#实现char字符数组与字符串相互转换的方法,结合实例形式简单分析了C#字符数组转字符串及字符串转字符数组的具体实现技巧,需要的朋友可以参考下
    2017-02-02
  • 浅谈C#多线程下的调优

    浅谈C#多线程下的调优

    本文主要介绍了C#多线程下的调优,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-04-04
  • 对WPF中的TreeView实现右键选定

    对WPF中的TreeView实现右键选定

    这篇文章介绍了WPF实现右键选定TreeView的方法,文中通过示例代码介绍的非常详细。对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-06-06
  • C#日期转换函数分享

    C#日期转换函数分享

    这篇文章介绍了C#日期转换函数,有需要的朋友可以参考一下
    2013-10-10
  • C#版Windows服务安装卸载小工具

    C#版Windows服务安装卸载小工具

    这篇文章主要为大家推荐了一款C#版Windows服务安装卸载小工具,小巧灵活的控制台程序,希望大家喜欢,感兴趣的小伙伴们可以参考一下
    2016-07-07
  • Unity调用C++ dll实现打开双目相机

    Unity调用C++ dll实现打开双目相机

    这篇文章主要为大家详细介绍了如何在Unity中调用C++ dll实现打开双目相机的功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2022-05-05

最新评论