转自:http://www.cnblogs.com/michaelxu/archive/2008/03/31/1131500.html

什么是委托
  首先要知道什么是委托,用最通俗易懂的话来讲,你就可以把委托看成是用来执行方法(函数)的一个东西。

如何使用委托
  在使用委托的时候,你可以像对待一个类一样对待它。即先声明,再实例化。只是有点不同,类在实例化之后叫对象或实例,但委托在实例化后仍叫委托。

声明,如:

1    namespace Vczx.ProCSharp.Exc
2    {
3        delegate double MathsOp( double x );
4        //class defination here
5    }

  这就声明了一个委托,意义:任何一个返回值为double,且只有一个形参为double的函数,都可以用这个委托来调用。
  注意:委托的声明位置在namespace里面,类的外面。其实,委托的声明也可以在类的里面,甚至是任何一个可以声明类的地方。
  实例化:
  首先我们要先有一个满足委托声明的方法,假设一个返回一个数的2倍的方法:

1class MathsOperations
2{
3    public static double MultiplyBy2( double value )
4    {
5        return value * 2;
6    }

7}

  有了这样一个方法,我们就可以实例化一个委托了:
MathsOp operation = new MathsOp( MathsOperations.MultiplyBy2 );
  在实例化一个委托时,要给它一个参数,这个参数就是委托执行的方法,它可以是静态方法,也可以是实例方法(这一点有别于函数指针,函数指针只能调用静态
方法),如:
MathsOp operation = new MathsOp( new Class1().Method1 );

在实例化完一个委托之后,就可以用这个委托来调用方法了:
double result = operation( 1.23 );

例子代码:

 1namespace Vczx.ProCSharp.Exc
 2{
 3    delegate double MathsOp( double x );
 4    class Start
 5    {
 6        public class MyDelegate
 7        {
 8            public static double MultiplyBy2( double x )
 9            {
10                return x * 2;
11            }

12        }

13        [STAThread]
14        static void Main(string[] args)
15        {
16            MathsOp operation = new MathsOp( MyDelegate.MultiplyBy2 );
17            double x = 1.23;
18            double result = operation( x );
19            Console.WriteLine( "{0} multiply by 2 is {1}", x, result ); 
20            Console.Read();
21        }

22    }

23}

多路广播委托
   前面使用的委托只包含一个方法调用。调用委托的次数与调用方法的次数相同。如果要调用多个方法,就需要多次显示调用这个委托。其实委托也可以包含多个方
法,这种委托就是多路广播委托。多路广播委托派生于System.MulticastDelegate,它的Combine方法允许把多个方法调用链接在一起,我们可以通过+=来向委托添加调用方法,也可以用-=删除其中的调用方法。如:

 1namespace Vczx.ProCSharp.Exc
 2{
 3    public class MyDelegate
 4    {
 5        public static void MultiplyBy2( double value )
 6        {
 7            double result =  value * 2;
 8            Console.WriteLine( "Multiplying by 2: {0} gives {1}", value, result );
 9        }

10
11        public static void Squre( double value )
12        {
13            double result = value * value;
14            Console.WriteLine( "Squaring: {0} gives {1}", value, result );
15        }

16    }

17
18    delegate void MathsOp( double x );
19
20    class Start
21    {
22        [STAThread]
23        static void Main(string[] args)
24        {
25            MathsOp operation = new MathsOp( MyDelegate.MultiplyBy2 );
26            operation += new MathsOp( MyDelegate.Squre );
27            double x = 1.23;
28            operation( x );
29
30            operation -= new MathsOp( MyDelegate.MultiplyBy2 );
31            operation( x );
32            
33            Console.Read();
34        }

35    }

36}

输出:
Multiplying by 2: 1.23 gives 2.46
Squaring: 1.23 gives 1.5129
Squaring: 1.23 gives 1.5129

  注意,多路广播委托声明时必须返回void,否则返回值不知道应该送回什么地方。对此,我做了一个测试:如果不将委托的声明返回void,则返回值返回的是最后一个链入委托链的方法的返回值,编译不会出错。


为什么要用委托
  使用委托使程序员可以将方法引用封装在委托对象内。然后可以将该委托对象传递给可调用所引用方法的代码,而不必在编译时知道将调用哪个方法。与C或C++中的函数指针不同,委托是面向对象,而且是类型安全的。

 

  回复  引用  查看    
#1楼 2008-03-31 17:50 | jillzhang      
在使用委托的时候,你可以像对待一个类一样对待它
----------------------
委托原本就是类,它继承的是MuticastDelegate,在CLR编译时候,编译器遇到delegate也会创建一个新的MuticastDelegate类实例.

  回复  引用  查看    
#2楼 2008-03-31 17:53 | lovecherry      
注意:委托的声明位置在namespace里面,类的外面。
????
虽然文中有一点不确切的地方,分享精神值得鼓励阿

  回复  引用  查看    
#3楼 2008-03-31 17:53 | jillzhang      
其实原本根本不需要委托的,直接传方法函数的地址也可以,比如windows中的钩子,但非托管上的这些用法不是类型安全的,因为传的地址上的类型不确定,这样就有一定风险,而delegate解决了类型安全的问题.传递给委托的地址上必须和声明时的类型一致
  回复  引用  查看    
#4楼 2008-03-31 17:55 | jillzhang      
注意:委托的声明位置在namespace里面,类的外面。
--------------------
这种明显是错误的说法! 
委托就是类,能声明类的地方都能声明它
甚至有在方法参数中声明委托的
匿名可以使得类在任意位置声明!

  回复  引用  查看    
#5楼 2008-03-31 17:58 | lovecherry      
多路广播委托派生于System.MulticastDelegate

----------------

应该淡化多路委托这个概念吧,现在有纯单路委托吗?

  回复  引用  查看    
#6楼 2008-03-31 17:58 | jillzhang      
传递给委托的地址上必须和声明时的类型一致
---------------------
说一致有些误导,因为有两种情况特殊
因为返回值可以为声明返回值的派生类,而方法的参数可以为声明委托参数的父类

  回复  引用  查看    
#7楼 2008-03-31 18:04 | jillzhang      
多路委托
----------------
新鲜词!
委托就是委托,怎么还分多路单路呢! 
delegate继承MuticastDelegate,在MuticastDelegate中有三个字段,_target,_targetMethod,_invokeList;三个名称具体忘记了,但是这三个分别表示委托方法所在对象,委托方法地址,如果委托指向多个方法,invokeList就是多个方法的集合._target在静态委托的时候为null

  回复  引用  查看    
#8楼 2008-03-31 18:05 | jillzhang      
你所谓的多路就是
foreach( .. in invokeList)
{
Invoke(..);
}

  回复  引用  查看    
#9楼 2008-03-31 18:10 | Da Vinci      
委托就是类也不完全正确 委托是一种类似函数指针的东西
  回复  引用  查看    
#10楼 2008-03-31 18:13 | jillzhang      
@Da Vinci
委托就是类也不完全正确 委托是一种类似函数指针的东西
--------------------------------------------------------
在编译的时候,遇到delegate编译器会创建一个MuticastDelegate实例,它就是类。没有不完全正确的说法。

  回复  引用  查看    
#11楼 2008-03-31 18:22 | lovecherry      
@jillzhang
编译器做了很多工作,隐瞒了很多事实,真不知道是喜事悲

  回复  引用    
#12楼 218.205.227.*2008-03-31 18:27 | guest [未注册用户]
--引用--------------------------------------------------
jillzhang: 多路委托
----------------
新鲜词!
委托就是委托,怎么还分多路单路呢! 
delegate继承MuticastDelegate,在MuticastDelegate中有三个字段,_target,_targetMethod,_invokeList;三个名称具体忘记了,但是这三个分别表示委托方法所在对象,委托方法地址,如果委托指向多个方法,invokeList就是多个方法的集合._target在静态委托的时候为null
--------------------------------------------------------
应该是叫多点委托吧~~

  回复  引用  查看    
#13楼 2008-03-31 22:54 | Angel Lucifer      
delegate的本质是类是正确的,这样做的目的完全是为了类型安全。

所有的委托类型均继承自MulticastDelegate类,可以翻译成多播委托。

同时,Microsoft在委托的设计上有瑕疵,本来应该只有一个委托类,结果它给搞了两个,一个MulticastDelegate,另一个则是Delegate类。

  回复  引用  查看    
#14楼 [楼主]2008-04-01 07:30 | loose_went      
@jillzhang
是的,委托是一个类,也是一个派生于System.Delegate的一个类的实例。

  回复  引用  查看    
#15楼 [楼主]2008-04-01 07:32 | loose_went      
@jillzhang
是的,委托是类型安全的,这样在一定程度上可以避免一些因类型问题而产生的风险。

  回复  引用  查看    
#16楼 [楼主]2008-04-01 07:38 | loose_went      
@jillzhang
请教:
说一致有些误导,因为有两种情况特殊
因为返回值可以为声明返回值的派生类,而方法的参数可以为声明委托参数的父类
-----------------------
能详细解释一下吗?谢谢。

  回复  引用  查看    
#17楼 2008-04-01 09:15 | Tristan(Guozhijian)      
仔细读一下CLR VIA C#关于委托那一章基本上就清楚了
  回复  引用    
#18楼 61.187.8.*2008-04-01 09:45 | wmj [未注册用户]
说得比较通俗
  回复  引用  查看    
#19楼 2008-04-01 15:01 | 毅无涯      
楼主这样说是为了让大家更容易理解一些
  回复  引用  查看    
#20楼 2008-04-01 15:54 | jillzhang      
@Tristan(Guozhijian)
这本书是比较不错。能澄清很多疑难杂症

  回复  引用  查看    
#21楼 [楼主]2008-04-01 18:37 | loose_went      
@毅无涯
其实写这个东西只是自己学了之后的笔记而已,如果真要写成教别人的文章来,那一定得要在这个主题上有更深入的理解才行,才能用真正易懂的语言来表达。

  回复  引用  查看    
#22楼 2008-04-01 21:28 | airwolf2026      
我晕

ms-help://MS.MSDNQTR.v90.chs/dv_csref/html/97de039b-c76b-4b9c-a27d-8c1e1c8d93da.htm
这段链接是MSDN(local)上关于C#委托的编程指南,已经说的很清楚啦.而且还说到了一些注意点.
特别是 委托的"+-"操作,是创建了一个新对象.

  回复  引用    
#23楼 222.171.7.*2008-06-08 10:27 | qijund [未注册用户]
对我这种新手很有帮助的,我看msdn看书都费了很大精力还没理解,看了这篇文章,再回过头看msdn明白了很多。
大家做技术的严谨态度,小弟佩服。

  回复  引用    
#24楼 60.212.41.*2008-07-24 10:31 | 啊啊 [未注册用户]
多谢啊 我正为委托郁闷呢, LZ 转走啦 多谢 再次感谢
  回复  引用  查看    
#25楼 2008-08-02 19:08 | 孤独客      
讲得很简单明了
受用了

  回复  引用    
#26楼 221.219.181.*2008-08-11 11:52 | wxg [未注册用户]
c#没有指针,当然也没有函数指针。为了实现回调,所以设计了委托,当然委托的安全性好于函数指针。
  回复  引用    
#27楼 59.126.250.*2008-09-09 16:59 | 烽 [未注册用户]
樓主不好意思,要糾正您的中文了。
"首先我们要先有一个满足委托声明的方法,假设一个返回一个数的2倍的方法"
在上面的一句話的中"2倍"你弄錯了。
假設數字是1,那中文的1倍是2(1+1),2倍就是3(1+1+1)。
所以依文你應是乘3

1class MathsOperations
2{
3 public static double MultiplyBy2( double value )
4 {
5 return value * 3;
6 }
7}

不然您應該改為
"首先我们要先有一个满足委托声明的方法,假设一个返回一个数的1倍的方法"

  回复  引用  查看    
#28楼 [楼主]2008-09-11 04:05 | loose_went      
@烽
想必楼上都这个倍数的用法还没有理解。
一倍就是乘1,两倍就是乘2,如果说多一倍,那么就是乘2,多两倍就是乘3。
对于数字1,应另当别论,人们习惯的用法是,1的一倍就是2,但1的两倍也是2,也就是说1的一倍比1的1.5倍还大,这根本不合逻辑,但人们已经习惯这么用了。实际上一倍就是乘1,2倍就是乘2,没有什么好讲的。

  回复  引用    
#29楼 59.126.250.*2008-09-11 09:40 | 烽 [未注册用户]
唉~~現在多是積非成是之人,明明是錯的,想別人也一樣就一樣吧。
自己明明是在用中文,還硬凹說數字是別當別論,把自己的文化當成甚麼了。
倍就是倍,乘就是乘。

所以根本是自己用錯,您說的
"1的一倍就是2,但1的两倍也是2,也就是说1的一倍比1的1.5倍还大"
根本是
"1的乘二就是2,1的一倍也是2,也就是说1的一倍比1的乘1.5还大,這是合理的"
這才對,所以是您自己把中文小看了、亂看了。

自己本是心存感激(因看了樓主的文章覺得簡明易懂,幫助甚大),還看了樓主自己寫的對聯,但現在覺得也只有編程可看而已。

  回复  引用    
#30楼 59.126.250.*2008-09-11 09:45 | 烽 [未注册用户]
不然你以為中文為甚會有倍和乘,老祖宗糊涂了嗎?
還是吃飽有空,又發明了一個一樣意思的"字"。

  回复  引用  查看    
#31楼 [楼主]2008-09-12 01:56 | loose_went      
汉典对"倍"是这样解释的:
● 倍

bèi

◎ 等于原数的两个:加~。事~功半。~道而行(兼程而行)。

◎ 某数的几倍等于用几乘某数:二的五~是十。

◎ 更加,非常:“每逢佳节~思亲”。~加。~儿精神。

◎ 增益:“焉用亡郑以~邻?”

◎ 古同“背”,背弃,背叛。

◎ 古同“背”,背诵。

1、等于原数的两个,但请注意,加倍、事半功倍、倍道而行中的倍都有两倍的意思,而不是一倍。
2、某数的几倍等于用几乘某数(应该知道是什么意思了吧?)

  回复  引用    
#32楼 59.126.250.*2008-09-12 16:50 | 烽 [未注册用户]
都說積非成是了,您往古文找,有現在這樣的用法嗎?

◎ 某数的几倍等于用几乘某数:二的五~是十。 <-- 這就是積非成是所造成的

"1、等于原数的两个,但请注意,加倍、事半功倍、倍道而行中的倍都有两倍的意思,而不是一倍"
兩個就是該東西的一倍,事半功倍就是多了一倍的功夫。

這是我在中國台灣的國語詞典看到的,我只取古文

@違背、反叛。說文解字:「倍,反也。」禮記˙緇衣:「信以結之,則民不倍。」
@背向、背著。戰國策˙趙策三:「天子弔,主人必將倍殯柩。」史記˙卷九十二˙淮陰侯傳:「兵法右倍山陵,前左水澤。」
@增加與原數相等的數。孟子˙公孫丑上:「故事半古之人,功必倍之。」南朝梁˙劉勰˙文心雕龍˙鎔裁:「雖翫其采,不倍領袖。」

@更加、益增。如:「勇氣倍增」、「倍受恩寵」。唐˙王維˙九月九日憶山東兄弟詩:「獨在異鄉為異客,每逢佳節倍思親。」


算了,您有您的堅持,我有我的想法。

文學上無法交流,就只交流編程吧~~。

  回复  引用  查看    
#33楼 [楼主]2008-09-12 17:59 | loose_went      
十分敬佩老兄的这种精神,这正是一个搞技术的人应该必备的。
  回复  引用    
#34楼 122.225.11.*2008-09-23 23:12 | seesky [未注册用户]
挺通俗易懂的,挺好,能明白就好
  回复  引用    
#35楼 59.41.16.*2008-09-26 12:45 | 中古 [未注册用户]
在看MSDN的时候这些概念看得满头雾水,但看完这篇文章后一下子就明白了
概念的东西还是用自己的经验来描述容易理解,虽然有时候不是那么严谨,但对新手入门来说却大有帮助(概念就是要解释给新手理解的)

  回复  引用  查看    
#36楼 2008-10-14 10:19 | Goumh      
委托这东西,看官方文章,实在是太难理解,还是LZ的容易理解,谢谢了。
  回复  引用    
#37楼 59.174.196.*2008-10-19 11:21 | 金钱豹 [未注册用户]
谢谢楼主!
讲的非常通俗易懂!
把一件难的事情讲得容易理解是很不容易的!
有些朋友就不要钻牛角尖了!

  回复  引用  查看    
#38楼 2008-10-20 11:42 | 非法操作      
比较通俗,大家的严谨态度很好,感谢楼主!
  回复  引用    
#39楼 24.18.203.*2008-12-22 12:41 | snowampl [未注册用户]
谢谢楼主,看了几部书,英文中文都有,都没有楼主说的明白,真是豁然开朗啊,非常非常感谢!
  回复  引用  查看    
#40楼 2009-01-13 02:02 | .NET钉子户      
话不多,看了一半,先说声:谢谢!!!
  回复  引用  查看    
#41楼 [楼主]2009-01-14 09:21 | loose_went      
只是简要的说明了一下个人的理解,其实委托多数是和事件一起使用的,关于事件,可以看一下http://www.cnblogs.com/michaelxu/archive/2008/04/02/1134217.html
  回复  引用  查看    
#42楼 2009-02-12 14:01 | cloudyy      
LZ这个例子 不错 受教了
  回复  引用    
#43楼 219.135.214.*2009-02-25 09:27 | liys [未注册用户]
还不错,谢!
  回复  引用  查看    
#45楼 2009-04-27 14:47 | 龙潜冰风悄林      
class Program
{
private delegate string GetAString();
static void Main(string[] args)
{

int x = 40;
GetAString firstStringmethods = new GetAString(x.ToString);
Console.WriteLine(firstStringmethods());
x = 80;
Console.WriteLine(firstStringmethods());
Console.WriteLine(x.ToString());
Console.ReadKey();
}
}

楼主,能不能解释下以上代码的运行结果,谢谢

posted on 2009-06-08 22:58  钱途无梁  阅读(595)  评论(0编辑  收藏  举报