zoukankan      html  css  js  c++  java
  • 委托与事件

    1. 委托定义

    委托(Delegate)是C#或者.NET中表示强类型方法的特殊类型。比较接近于C语言中的函数指针。(指向函数入口地址的数据类型)。读到这里说下C语言的两个概念:指针函数和函数指针。

    指针函数是指带指针的函数,即本质是一个函数。函数返回类型是某一类型的指针。

    int *fun(x);

     

    函数指针是指向函数的指针变量,即本质是一个指针变量。

    int (*f) (int x); /* 声明一个函数指针 */

    f=function; /* 将function函数的首地址赋给指针f */

     

    因此由上面可以看出,委托其实是一种封装好的函数指针。

     

    2. 申明委托

    申明委托使用delegate关键字,语法是:

    [<修饰符>] delegate <返回类型> <委托名> ([<形参表>])

    修饰符:new、public、protected、internal、private。如果在namespace下,只能用public,internal 。

    返回类型:与方法的返回类型相同

    委托名:与其他类型的标识符要求相同,常用handler作后缀

    形参表:与方法的形参表的要求相同

    举例:delegate void D1(int i, double d);

     

     

    3.委托的实例化

    由于委托类似方法指针,申明委托时给出的返回类型和形参表,实际上是指该委托所能指向或者封装的方法的返回类型和参数表。

    凡是与委托的返回类型和形参表相同的方法,不管是静态的还是实例的,称为与委托的类型兼容,都可以被该委托的实例封装。

    委托类型定义后,可以像其他类型一样申明该类型的变量,并对变量实例化。

     

    对委托进行实例化的写法为:

       new <委托类型名> (<表达式>)

       这里,表达式可以是一个兼容的静态方法、实例方法、或者一个委托实例。

    例如:

    delegate void D(int x);    //申明委托

    class C

    { public static void M1(int i) {...}

    public void M2(int i) {...}

    }

    class Test

    { static void Main()

    { D cd1 = new D(C.M1); // 用静态方法

    C t = new C(); // 实例化有关类

    D cd2 = new D(t.M2); // 用实例方法

    D cd3 = new D(cd2); // 用一个委托实例

    }

    }



    从某个方法创建一个委托实例时,该委托实例将封装此方法,此时,它的调用列表只包含一个进入点。 可用 +和 += 运算符组合委托实例,用 -和 -= 运算符将一个委托从委托组合中移除。

    当组合两个或多个非空委托实例时,它们的调用列表被连接在一起(按照左操作数在先、右操作数在后的顺序)以组成一个新的调用列表,它包含了两个或更多个“进入点”。

    例如:委托的组合及调用列表

     

    delegate void D(int x);

    class C

    { public static void M1(int i) {...}

    public static void M2(int i) {...}

    }

    class Test

    { static void Main()

    { D cd1 = new D(C.M1); // M1

    D cd2 = new D(C.M2); // M2

    D cd3 = cd1 + cd2; // M1 + M2

    D cd4 = cd3 + cd1; // M1 + M2 + M1

    D cd5 = cd4 – cd1; // M1 + M2

    }

    }



    5.委托的调用

    直接使用委托实例名和一组符合要求的参数可以调用该委托。

    调用一个委托实例,就是依次调用其调用列表中的方法。使用的是同一组参数。

    委托实例可以多次出现在一个调用列表中。这种情况下,它每出现一次,就会被调用一次。从这样的调用列表中移除委托,实际上移除的是调用列表中最后出现的那个委托实例。

     

    6.匿名方法

    匿名方法是指初始化委托时内联申明的方法。

     

    7.委托程序实例 

    using System;

    namespace delegateExam
    {
    /// <summary>
    /// Class1 的摘要说明。
    /// </summary>
    delegate void TestDelegate(); //在名称空间下申明委托
    class Class1
    {
    /// <summary>
    /// 应用程序的主入口点。
    /// </summary>
    TestDelegate TD; //申明委托类型的字段
    [STAThread]
    static void Main(string[] args)
    {
    //
    // TODO: 在此处添加代码以启动应用程序
    //
    TestDelegate TD1 = new TestDelegate(Class1.MethodA); //在方法中申明委托实例并用静态方法初始化
    TestDelegate TD2 = new TestDelegate(test.method2); //在方法中申明委托实例并用静态方法初始化
    Class1 aClass1 = new Class1();
    test aTest = new test();
    TD2 += new TestDelegate(aClass1.MethodB); //加上实例方法
    TD1 += new TestDelegate(aTest.method1); //加上实例方法
    Console.WriteLine("调用委托TD1...");
    TD1();
    Console.WriteLine("调用委托TD2...");
    TD2();
    TD1 += TD2; //合并两个委托的调用列表
    Console.WriteLine("调用委托TD1和TD2...");
    TD1();
    TD1 -= TD2;
    Console.WriteLine("移除TD2..."); //移除委托TD2
    TD1();
    TD1 += delegate { Console.WriteLine("添加匿名方法..."); };//添加匿名方法
    TD1();
    Console.WriteLine("赋值给委托类型的字段...");
    aClass1.TD = TD2; //给字段赋值
    aClass1.TD();
    Console.ReadLine();

    }
    static void MethodA()
    {
    Console.WriteLine("类Class1的静态方法MethodA.");
    }
    public void MethodB()
    {
    Console.WriteLine("类Class1实例方法MethodB.");
    }
    }
    class test
    {
    public void method1()
    {
    Console.WriteLine("类test的实例方法method1.");
    }
    public static void method2()
    {
    Console.WriteLine("类test的静态方法method2.");
    }
    }

    }

     

    8.事件定义

    事件也是类的成员之一,也是一个特殊的委托类型。相当于一个类型为委托的字段或属性;

    事件是当发生某些事情时,类向该类的使用者(其他类)提供通知的一种方法。

     

    9.事件的申明

    C#中申明事件的格式之一为:

    <事件修饰> event <委托类型> <事件名称>;

    事件修饰可以为:new,public,protected,internal,private,Static,virtual,Sealed,override,abstract,extern

    例如:

    public event ChangedEventHandler Changed;

    这里,ChangedEventHandler是委托类型。必须事先申明,且必须至少具有与事件本身一样的可视性。 如:

    public delegate void ChangedEventHandler(object sender, EventArgs e);



    声明事件的方法与声明委托类型的字段类似,只是关键字 event 在事件声明前面,在修饰符后面。事件通常被声明为公共事件,但允许任意可访问修饰符。

     

    10.事件的调用

    在定义事件的类里面,可以像使用字段那样使用事件(检查其值和给其赋值);

    在发生需要通知客户类的事情时,调用事件。由于事件本质上是委托类型,调用事件与调用委托一样。写法为:

        <事件名>([<实参表>]);

    例如:If (Changed != null)  Changed(this,e);

     

    11.事件订阅与移除

    在定义事件的类外部,只能将委托实例添加到事件的调用列表中或从中移出。写法同对委托的操作。例如:

             List.Changed += new ChangedEventHandler(ListChanged);

    或    

             List.Changed -= new ChangedEventHandler(ListChanged);

    重要换句话说,在类外部不可以调用事件,或像使用变量一样使用事件

    这就是事件不同于委托的地方,也是不能完全用委托代替事件的原因。

    12.委托程序实例 

    using System;

    //下面的程序展示了标准委托的定义、事件的定义、事件处理方法的定义及他们的综合应用
    //程序将列出0到100之间的所有偶数
    namespace eventExam
    {
    //定义委托类型
    delegate void EvenNumberHandler(object sender, OnEvenNumberEventArgs args);

    //定义参数类
    public class OnEvenNumberEventArgs : EventArgs
    {
    private int EvenNumber; //存放偶数的字段
    public OnEvenNumberEventArgs(int evenNumber) //构造方法
    {
    this.EvenNumber = evenNumber;
    }
    public int Number //返回当前偶数的只读属性
    { get { return EvenNumber; } }
    }

    //包含有事件的类
    class Counter
    {
    public event EvenNumberHandler OnEvenNumber; //定义事件
    public Counter()
    {
    OnEvenNumber = null; //在构造方法中,给事件赋初值。只有在定义事件的类中可以这样。
    }
    //触发事件的方法
    public void CountTo100()
    {
    int CurrentNumber;
    for(CurrentNumber=0;CurrentNumber<=100;CurrentNumber++)
    {
    if(CurrentNumber % 2 == 0) //如果为偶数
    {
    if (OnEvenNumber != null) //如果事件的调用列表不是空的
    {
    OnEvenNumberEventArgs theArgs = new OnEvenNumberEventArgs(CurrentNumber);
    OnEvenNumber(this, theArgs); //触发事件(其实就是调用委托)
    }
    }
    }
    }
    }

    //对事件进行处理的类
    class EvenNumberHandlerClass
    {
    //对事件进行处理的方法
    public void EvenNumberFound(object sender, OnEvenNumberEventArgs args)
    {
    Console.Write(args.Number);
    Console.Write("; ");
    }
    }
    /// <summary>
    /// Class1 的摘要说明。
    /// </summary>
    class MainClass
    {
    /// <summary>
    /// 应用程序的主入口点。
    /// </summary>
    [STAThread]
    static void Main(string[] args)
    {
    Counter MyCounter = new Counter(); //申明并初始化事件发生的对象
    //申明并初始化事件处理对象
    EvenNumberHandlerClass MyEvenNumberHandlerClass = new EvenNumberHandlerClass();
    //将事件发生对象的事件安装上事件处理方法。写法同对委托变量的处理。
    MyCounter.OnEvenNumber += new EvenNumberHandler(MyEvenNumberHandlerClass.EvenNumberFound);
    MyCounter.CountTo100();
    Console.ReadLine();
    }
    }
    }



    13.标准化事件的设计

    事件可以是系统定义的委托类型,也可以是自己定义的任何委托类型;

    但最好采用标准的委托类型;

    标准的事件委托类型有两个参数。

    引发事件的对象(命名为:sender)

    包含与事件有关数据的参数对象

             注意,第二个参数包含所有事件数据。因此需要自己根据需要定义。定义时必须继承系统定义的EventArgs类。

     

    事件参数类定义举例:

    public class OnEvenNumberEventArgs : EventArgs

    {

    private int EvenNumber;

    public OnEvenNumberEventArgs(int evenNumber)

    {

    this.EvenNumber = evenNumber;

    }

    public int Number

    { get { return EvenNumber; } }

    }



    14.在程序中定义和使用事件的过程:

    1.定义委托类型;

    2.定义事件参数类型;

    3.在类中定义事件;

    4.编写事件处理方法(必须符合委托要求);

    5.用事件处理方法初始化委托实例;

    6.将委托实例添加到事件的调用列表;

    7.触发事件。

    完….待续!

    .NET Framework

     
     


  • 相关阅读:
    gridview展示行号
    DateEdit如果开启Vista模式并显示日期+时间模式
    DevExpress GridView 添加和设置右键菜单
    C# WinForm实现粘贴图片到PictureBox及复制PictureBox中的图片
    dll反编译工具(ILSpy)的使用
    Dev的双击Gridview的DoubleClick
    SQL Server日期时间格式转换字符串详解
    LabelControl文本居中显示
    C# winform 判断click事件点击的是左键还是右键
    Winform窗体状态的判断及调用打开的窗体的方法
  • 原文地址:https://www.cnblogs.com/Leo_wl/p/2363214.html
Copyright © 2011-2022 走看看