原文链接:http://www.cnblogs.com/kissazi2/p/3189685.html
从本篇文章中,我们将从代码层面知道委托和事件的差异和用处,并且我们不会用猫和老鼠之类奇怪的比喻来混淆大家。
首先,我们知道委托和事件都可以用来调用跟自己方法签名一样的方法。容易混淆大家的地方大多在于这两者之间的区别。从我们使用委托delegate和事件event上的主要有两大区别(如表1所示)。
表1 委托和事件的区别
序号 | 区别 | 委托 | 事件 |
1 | 是否可以使用=来赋值 | 是 | 否 |
2 | 是否可以在类外部进行调用 | 是 | 否 |
3 | 是否是一个类型 | 是 | 否,事件修饰的是一个对象 |
一个例子
我们将创建两个类ClassA和ClassB,这两个类都很简单,只有一个方法,并且这两个类的方法签名一样。ClassC内有委托和事件,为了演示方便,我们将委托和事件的访问权限都设为public。下面我们将主要看看委托和事件在使用上面的区别。
ClassA
class ClassA
{
public static void ClassA_Say()
{
Console.WriteLine("ClassA_Say");
}
}
ClassB
class ClassB
{
public static void ClassB_Say()
{
Console.WriteLine("ClassB_Say");
}
}
ClassC
class ClassC
{
public delegate void Say();
public static event Say Say_EventHandler;
private static int myVar;
public static int MyProperty
{
get { return myVar; }
set
{
myVar = value;
if (Say_EventHandler != null)
{
Say_EventHandler();
}
}
}
}
区别1:委托是一个类型,事件修饰的是一个对象
class Class1
{
static void Main(string[] args)
{
//正确使用
ClassC.Say handle = ClassA.ClassA_Say;
//编译器报错
ClassC.Say_EventHandler handle2 = ClassB.ClassB_Say;
}
}
错误:“ClassC.Say_EventHandler”是“字段”,但此处被当做“类型”来使用
区别2:委托可以在声明它的类外部进行调用,而事件只能在类的内部进行调用。
(1)在类外部调用委托
static void Main(string[] args)
{
ClassC.Say handler = ClassA.ClassA_Say;
handler += ClassB.ClassB_Say;
handler();
}
结果截图:
(2)在类外部调用事件
static void Main(string[] args)
{
ClassC.Say_EventHandler += ClassA.ClassA_Say;
ClassC.Say_EventHandler += ClassB.ClassB_Say;
//编译器提示错误
ClassC.Say_EventHandler();
}
提示错误:
事件“ClassC.Say_EventHandler”只能出现在 += 或 -= 的左边(从类型“ClassC”中使用时除外)
从编译器提示的错误,我们可以了解到,事件只能在声明它的类内部被调用。从事件本身来讲,事件一般用于类自身的属性变化时,用来通知外界自身的变化的。我们将对ClassC内部的一个属性赋值,然后调用事件,模拟对外通知。代码如下所示
static void Main(string[] args)
{
ClassC.Say_EventHandler += ClassA.ClassA_Say;
ClassC.Say_EventHandler += ClassB.ClassB_Say;
ClassC.MyProperty = 1;
}
结果截图
总结:
1、委托可以把一个方法作为参数代入另一个方法。 委托可以理解为指向一个函数的指针。
2、委托和事件没有可比性,因为委托是类型,事件是对象
3、其实事件是委托的一个修饰符,加了event(事件)修饰之后,委托就被阉割了,假设这个加了修饰符的事件称为事件A,通过事件A我们可以更好地控制注册和注销,也就是对于一个外部类,那么他只能“注册自己+=、注销自己-=”,例如上面的ClassA,如果我们事件的注册是在ClassA内部进行的,那么它就只能注册自己内部的方法到事件上。另外,外界也不能主动地触发一个事件。事件只能add、remove自己,不能赋值。也就是说事件只能+=、-=,不能= 。当我们反编译以上例子的源程序的时候,我们可以发现事件内部就是一个private的委托和add、remove两个方法 (如下图所示), 在add_Say_EventHandler()方法内部,实际上调用了System.Delegate的Combine()静态方法,这个方法用于将当前的变量添加到委托链表中。
在我们定义委托的时候:
public delegate void GreetingDelegate(string name);
当编译器遇到这段代码的时候,会生成下面这样一个完整的类:
public sealed class GreetingDelegate:System.MulticastDelegate{
public GreetingDelegate(object @object, IntPtr method);
public virtual IAsyncResult BeginInvoke(string name, AsyncCallback callback, object @object);
public virtual void EndInvoke(IAsyncResult result);
public virtual void Invoke(string name);
}