C#--委托和事件的区别

原文链接:http://www.cnblogs.com/kissazi2/p/3189685.html

从本篇文章中,我们将从代码层面知道委托和事件的差异和用处,并且我们不会用猫和老鼠之类奇怪的比喻来混淆大家。

首先,我们知道委托和事件都可以用来调用跟自己方法签名一样的方法。容易混淆大家的地方大多在于这两者之间的区别。从我们使用委托delegate和事件event上的主要有两大区别(如表1所示)。

                          表1 委托和事件的区别

序号区别委托事件
1是否可以使用=来赋值
2是否可以在类外部进行调用
3是否是一个类型否,事件修饰的是一个对象

--参考MSDN上关于委托事件的定义。

一个例子

我们将创建两个类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();
        }

结果截图:

image

(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;
        }

结果截图

image

 

总结:

1、委托可以把一个方法作为参数代入另一个方法。 委托可以理解为指向一个函数的指针。

2、委托和事件没有可比性,因为委托是类型,事件是对象

3、其实事件是委托的一个修饰符,加了event(事件)修饰之后,委托就被阉割了,假设这个加了修饰符的事件称为事件A,通过事件A我们可以更好地控制注册和注销,也就是对于一个外部类,那么他只能“注册自己+=、注销自己-=”,例如上面的ClassA,如果我们事件的注册是在ClassA内部进行的,那么它就只能注册自己内部的方法到事件上。另外,外界也不能主动地触发一个事件。事件只能add、remove自己,不能赋值。也就是说事件只能+=、-=,不能= 。当我们反编译以上例子的源程序的时候,我们可以发现事件内部就是一个private的委托和add、remove两个方法 (如下图所示), 在add_Say_EventHandler()方法内部,实际上调用了System.Delegate的Combine()静态方法,这个方法用于将当前的变量添加到委托链表中。

image

在我们定义委托的时候:

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);
 }

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值