原文地址:http://www.cnblogs.com/qingyuan/archive/2010/05/11/1732415.html

1.什么是委托,为什么要使用委托

我正在埋头苦写程序,突然想喝水,但是又不想自己去掉杯水而打断自己的思路,于是我就想让女朋友去给我倒水。她去给我倒水,首先我得让她知道我想让她干什么,通知她之后我可以继续写自己的程序,而倒水的工作就交给了她。这样的过程就相当于一个委托。

在程序过程中,当程序正在处理某个事件的时候,我需要另外的程序代码去辅助处理一些事情,于是委托另一个程序模块去处理,而委托就可以达到这种目的,我可以利用委托通知另外的程序模块,该去调用哪个函数方法。委托其实就起到了这样一个作用,将函数签名传递到了另一个函数中。或许这样讲还是有些模糊,看看后面的具体实例。

2.委托的定义

delegate int Add(int num1,int num2);

delegate void ConvertNum(string result);

上面是定义两个委托的例子,其实很简单。声明一个委托使用delegate关键字,上面分别是定义的带返回值的委托和不带返回值的委托,

两个委托都有传递参数,当然也可以不传递参数。其实委托也是一个类,委托派生为System.MulticastDelegate,而System.MulticastDelegate

又继承System.Delegate,如果你知道这个也就明白委托其实是一个特殊的类。

 public delegate string TeaDelegate(string spText);

     public class DelegateSource
{
public void TestDelegate()
{
Operator op = new Operator();
TeaDelegate tea = new TeaDelegate(op.GetTea);
Console.WriteLine("去给我倒杯水");
Console.WriteLine();
string result=tea("去给我倒杯水");
Thread.Sleep();
Console.WriteLine(result);
Console.WriteLine();
}
} public class Operator
{
/// <summary>
/// 确定是否还有水
/// </summary>
private bool flag = true; public string GetTea(string spText)
{
if (spText == "去给我倒杯水")
{
if (flag)
{
return "老公,茶来了";
}
else
{
return "老公,没有水了";
}
}
return "等待.......";
}
}

输出结果

上面使用最普通的一种方式来定义了一个委托的使用,这个例子虽然很简单,但是能够很形象的描述委托的使用。

3.委托的三种形式

(1).推断

 public delegate string TeaDelegate(string spText);

     public class DelegateSource
{
public void TestDelegate()
{
Operator op = new Operator();
TeaDelegate tea = op.GetTea;
Console.WriteLine("去给我倒杯水");
Console.WriteLine();
string result=tea("去给我倒杯水");
Thread.Sleep();
Console.WriteLine(result);
Console.WriteLine();
}
} public class Operator
{
/// <summary>
/// 确定是否还有水
/// </summary>
private bool flag = true; public string GetTea(string spText)
{
if (spText == "去给我倒杯水")
{
if (flag)
{
return "老公,茶来了";
}
else
{
return "老公,没有水了";
}
}
return "等待.......";
}
}

在委托定义的例子中我们看到委托的使用方法是在委托实例化的时候指定的[new DelegateName(FunctionName)],这里可能表述不是太但是代码应该看得白了。 而委托的推断,并没有new 委托这个步骤,而是直接将Function 指定给委托。

(2).匿名函数

  public delegate string TeaDelegate(string spText);

     public class DelegateSource
{
public void TestDelegate()
{
Operator op = new Operator();
bool flag = true;
TeaDelegate tea = delegate(string spText)
{
if (spText == "去给我倒杯水")
{
if (flag)
{
return "老公,茶来了";
}
else
{
return "老公,没有水了";
}
}
return "等待.......";
}; Console.WriteLine("去给我倒杯水");
Console.WriteLine();
string result=tea("去给我倒杯水");
Thread.Sleep();
Console.WriteLine(result);
Console.WriteLine();
}
}

至于匿名委托,给人的感觉更为直接了,都不用显示的指定方法名,因为根本没有方法,而是指定的匿名方法。匿名方法在.NET 中提高了

代码的可读性和优雅性。对于更多操作较少的方法直接写为匿名函数,这样会大大提高代码的可读性。这里有两个值得注意的地方: 第一,不能使用

跳转语句跳转到该匿名方法外,第二 不能使用ref,out修饰的参数

(3).多播委托

 public delegate string TeaDelegate(string spText);

     public class DelegateSource
{
public void TestDelegate()
{
Operator op = new Operator(); TeaDelegate tea1 = op.GetTea;
TeaDelegate tea2 = op.Speak;
TeaDelegate tea = tea1 + tea2; Console.WriteLine("去给我倒杯水");
Console.WriteLine();
string result=tea("去给我倒杯水");
Thread.Sleep();
Console.WriteLine(result);
Console.WriteLine();
}
} public class Operator
{
/// <summary>
/// 确定是否还有水
/// </summary>
private bool flag = true; public string GetTea(string spText)
{
if (spText == "去给我倒杯水")
{
if (flag)
{
return "老公,茶来了";
}
else
{
return "老公,没有水了";
}
}
return "等待.......";
} public string Speak(string spText)
{
Console.WriteLine("\n去把我的设计图稿拿来");
return null;
}
}

还是上面的那个实例,我不尽想让女朋友去给我掉杯水,还让她帮我将程序设计图稿拿过来。这个时候做的就不是一件事了,而是多件。

程序中也有很多这种情况,于是我们需要多播委托,在一个委托上指定多个执行方法,这是在程序中可以行的。上面提到了,委托直接继承于

System.MulticastDelegate,正是因为这个类可以实现多播委托。如果调用多播委托,就可以按顺序连续调用多个方法。为此,委托的签名就必须返回void;否则,就只能得到委托调用的最后一个方法的结果。所以在上面的这段代码中是得不到结果的

4.事件

使用C#编程,无论是WinForm,WebForm 给人很难忘得就是它的控件,而他们的控件库使用方式都是使用使用事件驱动模式,而事件驱动模式却少不了委托。话不多说,看代码能够更清好的理解事件和委托之间的联系.

 public delegate void MyDelegate(string name);

     public class EventSource
{
public event MyDelegate Event_Delegate; public void SetCustomer(string name)
{
Console.WriteLine("事件发生.....\n");
Console.WriteLine("hi! "+name);
} public void TestEvent()
{
EventSource source = new EventSource();
Console.WriteLine("订阅事件.....\n");
source.Event_Delegate += new MyDelegate(source.SetCustomer);
Console.WriteLine("触发事件.....\n");
source.Event_Delegate("hechen");
Console.WriteLine("..................");
}
}

上面的代码中我们定义了一个委托,然后定义了一个类EventSource,这个类中声明了一个事件。定义一个事件使用event 关键字,定义一

个event必须指定这个event传递消息的委托,在触发事件之前必需订阅事件,我们使用+= new 语法来订阅一个事件,也就相当于实例化一个事件。

当我们触发事件的时候,就会调用相应的方法去处理。

5. 泛型委托

委托是类型安全的引用,泛型委托就和我们常用的泛型类一样,这个类在使用的时候才能确定类型.通过泛型委托,我们可以在委托传递参数

之后知道它的类型.在.NET中有一个很典型的泛型委托:

public delegate voie EventHandler<TEventArgs>(object sender,TEventArgs e) where TEventArgs:EventArgs.

这是一个非常有特色的泛型委托,可能我们用的比较少,但是作用是不能忽视的。 我们看看三个非常具有代表性的泛型委托.现在.NET4.0已经出来了,但是泛型委托.NET2.0就出来了,Linq 大家用的那叫一个甜,

为啥 函数式编程风格,匿名方法,Lamda表达式表达式使用是如此的魅力。但是大家仔细观察过没有,Linq 中的方法有几个经常出现的参数:

Action<T>,Predicate<T>,Func<T, Result>

Func<T, E>:封装一个具有一个参数并返回 E 参数指定的类型值的方法,T 是这个委托封装方法的参数类型,E是方法的返回值类型。当然Func<T, Result> 只是其中的一种情况,这个委托还有其他的几种情况:Func<T> 这个是方法没有参数,返回值类型是T;Func<T1,T2,Result> 这个方法有两个参数,类型分别为T1,T2,返回值是Result,还有Func<T1,T2,T3,Result>,Func<T1,T2,T3,T4,Result> 这几中情况,具体情况就不介绍了.我们还可以通过扩展类型,扩展为更多的参数.

  public void TestFunc()
{
TEventSource eventSource=new TEventSource();
Func<string, string> func = eventSource.GetTea;
string result = func("茶");
Console.WriteLine(result);
} public string GetTea(string context)
{
if (context == "茶")
{
return "茶来了";
}
else
{
return "设计稿子来了";
}
}

Action<T>:封装一个方法,该方法只采用一个参数并且不返回值,包括Action<T>,Action<T1,T2>,Action<T1,T2,T3>,Action<T1,T2,T3,T4> 这几种情况,也可以通过扩展方法去扩展参数的个数 。

 public void TestAction()
{
TEventSource eventSource = new TEventSource();
Action<string> action = eventSource.Speak;
action("Action<T> 泛型委托");
} public void Speak(string context)
{
Console.WriteLine(context);
}

Predicate<T>:表示定义一组条件并确定指定对象是否符合这些条件的方法。该委托返回的是一个bool类型的值,如果比较满足条件

返回true,否则返回false.其实上面的Func 委托可以包含这个委托.不过这个委托和上面的两个不一样,它只有一种类型

 public void TestPredicate()
{
TEventSource eventSource = new TEventSource();
Predicate<int> predicate = eventSource.IsRigth;
Console.WriteLine(predicate());
} public bool IsRigth(int value)
{
if (value == )
{
return true;
}
else
{
return false;
}
}

6.异步委托

投票技术: 委托其实相当于一个线程,使用投票技术是使用异步委托的一种实现方式.Delegate类提供了方法BeginInvoke(),可以传送委托类型定义的输入参数,其返回类型为IAsyncResult。IAsyncResult的IsCompleted属性可以判断委托任务是否完成

  public delegate int DelegateVote(int data, int ms);
/// <summary>
/// 使用投票操作完成委托任务
/// </summary>
public class VoteDelegate
{
/// <summary>
/// 休眠特定时间执行操作
/// </summary>
/// <param name="data"></param>
/// <param name="ms"></param>
/// <returns></returns>
public static int TakeWork(int data, int ms)
{
Console.WriteLine("开始调用TakeWork方法");
Thread.Sleep(ms);
Console.WriteLine("结束调用TakeWork方法");
return data + ;
} public void TestDelegate()
{
DelegateVote voteDel = TakeWork;
IAsyncResult result = voteDel.BeginInvoke(,,null,null);
while (result.IsCompleted == false)
{
Console.WriteLine("等待......");
Thread.Sleep();
}
int value = voteDel.EndInvoke(result);
Console.WriteLine("委托调用结果: "+value);
}
}

等待句柄:等待句柄是使用AsyncWaitHandle属性访问,返回一个WaitHandle类型的对象,它可以等待委托线程完成其任务。在这个参数中可以设置最大的等待时间。

  public delegate string WaitDelegate(string content);

     public class WaitHandlerDelegate
{
public void TestWaitHander()
{
WaitDelegate del = GetTea;
IAsyncResult ar = del.BeginInvoke("hechen", null, null);
while (true)
{
Console.Write(".");
if (ar.AsyncWaitHandle.WaitOne(, false))
{
break;
}
}
string result=del.EndInvoke(ar);
Console.WriteLine(result); } public static string GetTea(string content)
{
return "茶来了 "+content;
}
}

异步回调:这个方式和投票技术有点类似,不过在投票方式中BeginInvoke()方法第三个参数指定了一个方法签名,而这个方法参数接收IAsyncResult 类型的参数。

 public delegate string AsyDelegate(string content);

     public class AsyncresultDelegate
{
public void TestAsync()
{
AsyDelegate del = GetTea;
del.BeginInvoke("hechen", delegate(IAsyncResult ar) {
Thread.Sleep();
string result = del.EndInvoke(ar);
Console.WriteLine(result);
}, null);
for (int i = ; i < ; i++)
{
Console.WriteLine("等待.....");
Thread.Sleep();
}
} public static string GetTea(string content)
{
return "茶来了 " + content;
}
}

C# 委托知识总结的更多相关文章

  1. net .异步委托知识

    以前在编程中,异步用的比较少,导致C# 一些基础的 东西用法都不怎么熟悉,经常要用的时候在去查找资料比较被动,而已没真正里面理解起来,始终感觉不是自己的知识 (题外话) 首先委托关键字  Delega ...

  2. C# 委托知识总结【转】

    1.什么是委托,为什么要使用委托 我正在埋头苦写程序,突然想喝水,但是又不想自己去掉杯水而打断自己的思路,于是我就想让女朋友去给我倒水.她去给我倒水,首先我得让她知道我想让她干什么,通知她之后我可以继 ...

  3. C#之委托和事件

    我想,读者们可能看过一部电影叫<全民目击>,在电影中,富豪林泰婚期将至,准新娘却惨死地下停车场,林泰的富二代女儿林萌萌成为最大嫌疑人,林泰不惜重金聘请国内顶级律师周莉为女儿辩护,而公诉方却 ...

  4. [.NET] C# 知识回顾 - 委托 delegate (续)

    C# 知识回顾 - 委托 delegate (续) [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6046171.html 序 上篇<C# 知识回 ...

  5. [C#] C# 知识回顾 - 委托 delegate

    C# 知识回顾 - 委托 delegate [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6031892.html 目录 What's 委托 委托的属性 ...

  6. [C# 基础知识系列]专题一:深入解析委托——C#中为什么要引入委托

    转自http://www.cnblogs.com/zhili/archive/2012/10/22/Delegate.html 引言: 对于一些刚接触C# 不久的朋友可能会对C#中一些基本特性理解的不 ...

  7. C#基础知识之事件和委托

    本文中,我将通过两个范例由浅入深地讲述什么是委托.为什么要使用委托.委托的调用方式.事件的由来..Net Framework中的委托和事件.委托和事件对Observer设计模式的意义,对它们的中间代码 ...

  8. C#基础知识六之委托(delegate、Action、Func、predicate)

    1. 什么是委托 官方解释 委托是定义方法签名的类型,当实例化委托时,您可以将其实例化与任何具有兼容签名的方法想关联,可以通过委托实例调用方法. 个人理解 委托通俗一点说就是把一件事情交给别人来帮助完 ...

  9. C#知识体系(二)用案例来理解委托与事件

    上一篇博客讲到了LinQ和lambda的常用方法 还有很多我们未知但c#设计团队已经为我们封装好的类和方法.随着我们不断的熟悉C#语言,渐渐的就会接触到其他的知识点,委托.事件.反射.线程.同步,异步 ...

随机推荐

  1. 【Spring】12、Spring Security 四种使用方式

    spring security使用分类: 如何使用spring security,相信百度过的都知道,总共有四种用法,从简到深为:1.不用数据库,全部数据写在配置文件,这个也是官方文档里面的demo: ...

  2. 【CSS学习】--- 文本样式

    一.前言 CSS文本属性可以定义文本的外观.通过文本属性,可以定义文本的颜色.字符间距,对齐文本,装饰文本,对文本进行缩进,等等. CSS常用的文本属性目录: text-align 文本对齐属性 te ...

  3. HTML5是什么,以及优点和缺点

    HTML5是超文本标记语言HTML的第五次重大修改 HTML 5 的第一份正式草案已于2008年1月22日公布 2013年5月6日, HTML 5.1正式草案公布 HTML5的优缺点是什么?作为HTM ...

  4. Spring MVC异常处理 和 重定向传递数据

    1.异常处理介绍 Spring在web项目中,如果在请求处理时出现异常,那输出会是Servlet响应.这时异常需要以某种方式转换为响应. Spring将异常转换为响应的方式: a.特定的Spring异 ...

  5. 一些关于Viewport与device-width的东西~(转)

    内容转自 http://www.cnblogs.com/koukouyifan/p/4066567.html 非常感谢 口口一凡 为我们提供的这篇文章,受益匪浅,特地转到自己的博客收藏起来. 以下是原 ...

  6. WebAPI路由、参数绑定

    ​ 一.测试Web API a)测试Web API可以用来检测请求和返回数据是否正常,可以使用Fiddler.Postman等工具.以Fiddler为例,这是一个http协议调试代理工具,它能够记录客 ...

  7. spring ApplicationContext中Bean的生命周期

    AbstractApplicationContext Spring的AbstractApplicationContext是ApplicationContext的抽象实现类,该抽象类的refresh方法 ...

  8. 怎么查找Jenkins的个人api token

    程序中可变部分解释:其中server.build_job方法传入的参数channel为分渠道构建参数,也即jenkins job的参数,这个参数随不同的日常job不同是不同的,实际编写脚本的过程中这个 ...

  9. [20180408]那些函数索引适合字段的查询.txt

    [20180408]那些函数索引适合字段的查询.txt --//一般不主张建立函数索引,往往是开发的无知,使用trunc等函数,实际上一些函数也可以用于字段的查询.--//以前零碎的写过一些,放假看了 ...

  10. zTree 优秀的jquery树插件

    zTree 优秀的jquery树插件,文档详细,渲染快 使用方法: 1.引用zTree的js和css文件 <link href="~/Content/zTree_v3/css/zTre ...