解C#中的代理和事件(一)
突然写这篇文章,真的有点,是在做作的感觉,我想这并不是什么,难以
理解的东西,事实上,很多人都写过,而且,我保证至少不比我写的差.可是
还是觉得有必要提出来.因为要去正确的理解代理和事件是很有趣的,也是
很必要的.那么好吧,下面我就来讲讲,它们之间的关系.当然还会有些例子.
首先我想说说有关事件.事件顾名思义当然是windows程序收到的消息.
那么我举几个有关事件的例子,如鼠标移动.按下.之类的都是.那么代理呢?
很多人都说它看上去,就是想是一个受托管的函数指针.我觉得这种说法很
正确,事实如此.在MSDN上,我们经常可以看到,有关代理的"多路广播"这个词
我觉得不错,可是不好理解.但我会在下面详细讲解的.
代理:(有的书上也翻译成指代或委托.英文是这样一来的"Delegate")
我想很多刚接触C#的人,都会对代理产生兴趣的.事实上也是如此,不了解它,你就没办法
来做windows程序.和传统意义上的函数指针有所不同的是,代理在C#中是一种类型,这样
它看上去,更安全也,更符合OO精神.代理实际上所做的工作就是通过引用把函数包裹起来
并使函数有一个有效的返回值.不知道我这样说是否好理解.那么我举个例子,你去建造房子.
很显然,我是在说你所做的事情.那么建造房子就是代理,它指代了,你要做的事情,可是它并
没有去做任何事情,事实上是,在建造房子这个工作里,你做了,那么结果是什么?当然是建立
一座房子.是的,建造房子就是代理,而如何建造房子则是函数应该完成的工作.而建造的是什么样
的房子,则是返回值.还记得,我曾经说过,代理是一种类型吗?呵呵..我想你应该记得,因为,
那是很新颖的,至少当时我那么认为.好吧,让我们来看看名称空间System.Delagate,看见了吗?那
就代理类.
举个例子:
public delegate void GetString()//我申明了一个代理
现在我要用到它了如下;
int i=100;
GetString gs=new GetString(i.ToString);//这里我吧int的ToString方法
填入了一个代理中.看上去想构造函数.这就是常在书上看到的"名称等效的,而
不是结构等效的",我想看到这儿你还是不明白.那么,我再来一个代理
如下:
float j=0.0001;
GetString gs=new GetString(j.ToString);//瞧见了,int的ToString方法
和float的ToString方法的结构是不一样的,可是名称和类型的返回值和
参数都一样.现在,我想,你应该理解了吧.
可是,我们经常会在MSDN中看到,这样的句子.单路代理和多路广播.看上去,有点不好理解.
事实上,我开始看这样的句子,是有点头痛的.那么,我想例子是最好的解说方式.
Single Delegate:(单路代理)
从字面上,我们可以这样来理解,这个代理只是单单代理了一个函数的工作.那么好吧,让
我们来看看它是如何工作的.下面我就来定义一个这样的代理:
public delegate bool Myfun(string str,int i)
现在我再来写一个方法如下:
bool CompareStrToInt(string s,int i)
{
if(s.CompareTo(i.ToString())==0)
return true;
else
return false;
}
这个方法完成的工作很简单对吧,只是比较字符而已.那么和代理有什么关系呢?还记得
我说的话吗?代理就是在把动词名词化.代码如下:
Myfun mf=new (CompareStrToInt);
string s="10000";
int i=10000;
ConSole.WriteLine("Value="+mf(s,i));
输出结果:
Value=true
这就是单路代理.它只代理一个.好吧,也许你想看看复杂的例子,更有趣的在后面呢,
该是讨论多路广播的时候了.
多路广播:
一个代理同时代理几个方法.就是我们前面说到的那样,你去建造房子,现在要不仅仅是
建造住宅,还的去建造花园等等其它建筑物.可是它们都是在建造房子,传递的参数也相同
返回值的类型也相同都是房屋.那么我们为什么不找一个代理人来完成这样的任务呢?把
这些事物交由他一个人来完成不是可以节省我们很多的时间和金钱.是的我们可以那样做
System.MulticastDelegate 实际上在.net framework中你还可以找到这个类,多路代理
MSDN上翻译成多路广播.事实上它还重载了操作符+=.其实多路广播和单路代理在使用方法
上区别不大.你可以看下面的例子.
using System;
namespace Multi_castDelegate
{
/// <summary>
/// Summary description for Class1.
/// </summary>
class MyClassDelegate
{
/// <summary>
/// The main entry point for the application.
/// </summary>
public delegate string IntDelegate(string s);
}
}
using System;
namespace Multi_castDelegate
{
/// <summary>
/// Summary description for MyImplementingClass.
/// </summary>
public class MyClass
{
public MyClass()
{
}
public static string WriteString(string s)
{
Console.WriteLine("Writing string");
return "null";
}
public static string logString(string s)
{
Console.WriteLine("loging string");
return "null";
}
public static string TransmitString(string s)
{
Console.WriteLine("Transmitting string");
return "null";
}
}
}
The Main class:
using System;
using System.Threading;
namespace Multi_castDelegate
{
/// <summary>
/// Summary description for Test.
/// </summary>
public class Test
{
public static void Main()
{
MyClassDelegate.StringDelegate
Writer,Logger,Transmitter;
MyClassDelegate.StringDelegate
myDelegate;
Writer=new
MyClassDelegate.StringDelegate(MyClass.WriteString);
/// calling Writer
Writer("hello i am Writer just acting like Single cast");
Logger=new MyClassDelegate.StringDelegate(MyClass.logString);
///calling Logger
Logger("hello i am Logger just acting like Single-cast");
Transmitter=new MyClassDelegate.StringDelegate(MyClass.TransmitString);
///calling Transmitter
Transmitter("hello i am Transmitter just acting like Single-cast");
///here mydelegate used the Combine method of System.MulticastDelegate
///and the delegates combine
myDelegate=(MyClassDelegate.StringDelegate)System.Delegate.Combine(Writer,Logger);
myDelegate("used Combine");
///here Transmitter is also added using the overloaded form of Combine
myDelegate+=Transmitter;
myDelegate("Using Overloaded Form");
///now using the Remove method
myDelegate=(MyClassDelegate.StringDelegate)System.Delegate.Remove(myDelegate,Writer);
myDelegate("Without Writer");
///overloaded Remove
myDelegate-=Transmitter;
myDelegate("Without Transmitter");
System.Threading.Thread.Sleep(2300);
}
}
}
(上面的例子是在一个国外网站上找到的,觉得不错,就直接套用了.)
上面的例子重点是看那两个已经重载的操作符."-="和"+=".通过上面的例子,你可以清楚的
看到多路广播是如何一次代理多个方法的.当然你也可以删除掉那些你不想要的用"-="操作
符就可以了.(那么我将在下一篇讨论事件)