源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

对c#中委托的理解

  • 时间:2020-06-29 04:38 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:对c#中委托的理解
理解委托从一个简单的例子开始 金城武演的有部老电影叫《薰衣草》,里面有个情节大概是这样的:小金收客户的钱,然后代表客户去向不同的人Say I love you。 一开始他的客户都是中国人,只需要说中文,如下代码示例,很简单,支持所有中国客户:
[u]复制代码[/u] 代码如下:
public class LoveManager {     public void Love(string name)     {         Console.WriteLine("我爱你, {0}", name);     } }
[u]复制代码[/u] 代码如下:
class Program {     static void Main(string[] args)     {         LoveManager loveManager = new LoveManager();         loveManager.Love("张曼玉");     } }
执行结果:
[u]复制代码[/u] 代码如下:
我爱你, 张曼玉
我留意到后来电影里出现了外国客户,我想代码应该是这样:
[u]复制代码[/u] 代码如下:
//枚举,可扩展多语种 public enum Language {     English,     Chinese }
[u]复制代码[/u] 代码如下:
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);     } }
[u]复制代码[/u] 代码如下:
class Program {     static void Main(string[] args)     {         LoveManager loveManager = new LoveManager();         loveManager.Love("张曼玉", Language.Chinese);         loveManager.Love("Sophie Marceau", Language.English);     } }
执行结果:
[u]复制代码[/u] 代码如下:
我爱你, 张曼玉 I love you, Sophie Marceau
OK,现在张曼玉能听懂“我爱你”,Sophie Marceau能听懂“I love you”。虽然支持了英汉双语表白,但以后还有法国客户,葡萄牙客户,阿拉伯客户怎么办?每扩展一个语种除了添加这个语种“我爱你”的方法,还得扩展枚举,扩展LoveManager.Love(),确实有些繁琐。   C语言时代:指针 此时,不得不提到C语言中大名鼎鼎的指针。指针允许把一个函数的地址作为参数传递给另一个函数,这个特性在以后的各种高级语言中得到了扩展和加强。先看如下C代码:
[u]复制代码[/u] 代码如下:
#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; }
执行结果:
[u]复制代码[/u] 代码如下:
this is func1 this is func2
在.Net中能不能像C语言一样,把函数作为一个参数传递并且调用呢?
[u]复制代码[/u] 代码如下:
//这段代码并不能被执行,但如果在.Net中可以这样写的话问题就会简单很多 Love("张曼玉", LoveChinese); Love("Sophie Marceau", LoveEnglish);
.Net中更完美的解决方案:委托 在.Net中不但可以像C语言一样将函数作为参数传递,并且.Net提供了类型安全机制和更加强大的功能,如下提供了使用委托的完整代码示例:
[u]复制代码[/u] 代码如下:
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);         }     } }
执行结果:
[u]复制代码[/u] 代码如下:
我爱你, 张曼玉 I love you, Sophie Marceau
定义委托
[u]复制代码[/u] 代码如下:
public delegate void LoveDelegate(string name);
我们现在对委托做一个总结: 委托是一个类,它定义了方法的类型,使得可以将方法当作另一个方法的参数来进行传递,这种将方法动态地赋给参数的做法,可以避免在程序中大量使用If-Else(Switch)语句,同时使得程序具有更好的可扩展性。 在C#中委托使用特有的关键字 delegate 来定义,在delegate之后紧跟的是函数签名。为了确保类型安全,.Net中的委托要求函数具有相同的签名,比如 func(int p) 和func(string p)不能使用同一个委托,因为它们的参数类型不一样。 [img]http://files.jb51.net/file_images/article/201304/201304140002.jpg[/img] 通过ILDasm.exe可以发现,定义委托的那行代码实际在编译时会自动生成一个类,如果要还原这个类,代码会是这样:
[u]复制代码[/u] 代码如下:
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,凡是可以定义类的地方,都可以定义委托。 委托的构造函数
[u]复制代码[/u] 代码如下:
LoveManager loveManager = new LoveManager(); //编译不能通过,委托必须使用带有一个参数的构造函数 //LoveDelegate loveDelegate = new LoveDelegate(); LoveDelegate loveDelegate = new LoveDelegate(loveManager.LoveChinese); loveDelegate("吴剑");
与类不同的是,委托必须使用带有一个参数的构造函数。 委托推断语法
[u]复制代码[/u] 代码如下:
LoveManager loveManager = new LoveManager(); //等同于:LoveDelegate loveDelegate = new LoveDelegate(loveManager.LoveChinese); LoveDelegate loveDelegate = loveManager.LoveChinese; loveDelegate("吴剑");
委托与方法进行绑定 回到上面的例子,有一天一富二代找到小金,说钱不是问题,你去张曼玉楼下,用中文喊一遍,再用英文喊一遍。
[u]复制代码[/u] 代码如下:
static void Main(string[] args) {         LoveManager loveManager = new LoveManager();         //定义委托变量         LoveDelegate delegate1;         //变量初始化(用中文喊一遍)         delegate1 = loveManager.LoveChinese;         //绑定方法(用英文再喊一遍)         delegate1 += loveManager.LoveEnglish;         delegate1("张曼玉"); }
执行结果:
[u]复制代码[/u] 代码如下:
我爱你, 张曼玉 I love you, 张曼玉
我们可以用 += 将多个方法绑定到一个委托,也可以使用  -= 移除方法与委托的绑定。 匿名方法 客户的需求总是千变万化,一个客户跟小金说,我要跟曼玉表白,除了用中英文,能不能后面再给我加一句,曼玉一听到这句准会答应我。
[u]复制代码[/u] 代码如下:
LoveManager loveManager = new LoveManager(); LoveDelegate loveDelegate = loveManager.LoveEnglish; loveDelegate += loveManager.LoveChinese; loveDelegate += delegate(string name) {     Console.WriteLine("{0}, 还记得大明湖畔的夏雨荷吗?", name); }; loveDelegate("曼玉");
执行结果:
[u]复制代码[/u] 代码如下:
I love you, 曼玉 我爱你,曼玉 曼玉,还记得大明湖畔的夏雨荷吗?
针对这位特殊客户使用了匿名方法,不是每个人示爱的时候都会提到大明湖畔的夏雨荷,也就是这位特殊客户使用一次而以,所以没有必要定义一个独立的方法。使用匿名方法可以减少编码量,降低代码复杂度。 Lambda(λ)表达式 C# 3.0为匿名方法提供了Lambda表达式,如下代码执行结果与上面的示例完全一致:
[u]复制代码[/u] 代码如下:
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)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部