当前位置: 移动技术网 > IT编程>开发语言>c# > 理解C#中的事件

理解C#中的事件

2019年07月18日  | 移动技术网IT编程  | 我要评论
前面文章中介绍了委托相关的概念,委托实例保存这一个或一组操作,程序中将在某个特定的时刻通过委托实例使用这些操作。 如果做过gui程序开发,可能对上面的描述会比较熟悉。在g

前面文章中介绍了委托相关的概念,委托实例保存这一个或一组操作,程序中将在某个特定的时刻通过委托实例使用这些操作。

如果做过gui程序开发,可能对上面的描述会比较熟悉。在gui程序中,单击一个button会触发一个click事件,然后会执行一系列的操作,这一系列的操作就被存放在一个委托实例中。

接下来我们就看看事件。

使用委托中的问题

回到前面文章中苹果和富士康的例子,苹果将iphone的组装、包装和运输的工作全部委托给了富士康。

根据上面的描述,我们修改了一下代码,在apple这个类中加入一个订单属性,苹果只要接到新的订单,就发送一个通知给富士康,然后富士康就会执行一系列的操作了(组装、包装和运输)。

在主程序中,苹果将iphone的组装、包装和运输工作委托给了富士康,然后苹果每次收到订单,就会通过委托实例"verdortoassembleiphone"让富士康去执行一系列操作。

复制代码 代码如下:

class apple
{
    //声明委托类型
    public delegate void assembleiphonehandler(int num);
    public assembleiphonehandler verdortoassembleiphone;

    public void designiphone()
    {
        console.writeline("design iphone by apple");
    }

    private int ordernum;
    public int ordernum
    {
        get { return this.ordernum; }
        set
        {
            this.ordernum = value;
            if (verdortoassembleiphone != null)
            {
                verdortoassembleiphone(this.ordernum);
            }
        }
    }
}

class foxconn
{
    //与委托类型签名相同的方法
    public void assembleiphone(int num)
    {
        console.writeline("assemble {0} iphone by foxconn", num);
    }

    public void packiphone(int num)
    {
        console.writeline("pack {0} ipnone by foxconn", num);
    }

    public void shipiphone(int num)
    {
        console.writeline("ship {0} iphone by foxconn", num);
    }
}


class program
{
    static void main(string[] args)
    {
        apple apple = new apple();
        foxconn foxconn = new foxconn();
        apple.verdortoassembleiphone = new apple.assembleiphonehandler(foxconn.assembleiphone);
        apple.verdortoassembleiphone += new apple.assembleiphonehandler(foxconn.packiphone);
        apple.verdortoassembleiphone += new apple.assembleiphonehandler(foxconn.shipiphone);

        apple.ordernum = 100;
    
        console.read();
    }
}

下面我们看下这个例子实现中的问题:

如果用户在建立委托链的时候错误的使用了"="而不是"+=",那么委托链就断了
在主程序中,我们可以绕过设置订单这一步,直接调用委托实例"apple.verdortoassembleiphone(99);"来让富士康执行操作,这点是不合理的

复制代码 代码如下:

class program
{
    static void main(string[] args)
    {
        apple apple = new apple();
        foxconn foxconn = new foxconn();

        //创建委托实例
        apple.assembleiphone = new apple.assembleiphonehandler(foxconn.assembleiphone);
        apple.assembleiphone += new apple.assembleiphonehandler(foxconn.packiphone);
        apple.assembleiphone = new apple.assembleiphonehandler(foxconn.shipiphone);
        apple.designiphone();
       
        //委托实例的调用
        apple.verdortoassembleiphone(99);
    
        console.read();
    }
}

事件的出现

为了解决上面两个问题,出现了事件这个概念,我们要做的改变只是在声明委托实例的时候加上event关键字。

复制代码 代码如下:

public event assembleiphonehandler verdortoassembleiphone;

这时,上面两处有问题的代码就会在编译的时候报错了。   

上面的问题是解决了,但是event关键字作用是什么,事件跟委托有什么关系,"verdortoassembleiphone"怎么理解?

深入理解事件

其实,下面这个语句还是比较难理解的,看到的第一反应就是,委托跟事件到底是什么关系,event关键字跟委托类型"assembleiphonehandler"是什么关系。

复制代码 代码如下:

public event assembleiphonehandler verdortoassembleiphone;

其实,事件可以理解成一个委托的属性,通过对委托实例的封装来对委托实例的访问进行一些限制。下面我们通过il来查看一下这个个程序,得到下图。

下面我们就结合il的查看结果来分析事件到底是什么。在开始之前,相信大家一定都熟悉属性(property)这个概念吧,那就让我们从熟悉的属性开始分析。

c#属性的概念

首先来看看我们熟悉的东西,"ordernum"是我们定义的一个订单数量的属性(property),它有一组get/set方法。通过这个属性的get/set方法,我们可以访问"ordernum"字段(field)。

通过il查看"ordernum"这个属性,我们可以看到这个属性对应的get/set方法。

复制代码 代码如下:

public int32 get_ordernum() { }
public void set_ordernum(int32 'value') { }

事件

接下来再看"verdortoassembleiphone"事件,我们按照属性的方式去理解事件,从il的截图中,可以看到事件中有一对addon/removeon方法来操作我们的委托实例(想想属性的get/set方法)。

同时,我们看到编译器给我们生成了一个private的field(如下),从这里我们可以看到事件"verdortoassembleiphone"本质上就是一个委托类型的变量。由于这个变量是private的,也就解释了为什么我们在定义事件的类外部不能直接访问这个变量。

复制代码 代码如下:

.field private class _1_1_delegate.apple/assembleiphonehandler verdortoassembleiphone

事件代码的转换
根据上面的分析,我们可以看到,编译器帮我们进行了下面的代码转换。这样一来,就相当于通过event关键字对委托实例访问增加了一些限制。

原始的声明事件的c#代码:

复制代码 代码如下:

public event assembleiphonehandler verdortoassembleiphone;

编译器转换后的代码:

复制代码 代码如下:

private assembleiphonehandler verdortoassembleiphone;

public void add_verdortoassembleiphone(assembleiphonehandler 'value') { }

public void remove_verdortoassembleiphone(assembleiphonehandler 'value') { }


通过上面的分析,可以了解到事件封装了委托类型的实例,使得:

1.在定义事件的类内部,不管你声明它是public还是protected,它总是private的;也就是说在在定义事件的类外部不能对事件进行调用
2.在定义事件的类外部,添加"+="和移除"-="委托实例的访问限定符与声明事件时使用的访问符相同
2-1).也就是说,如果事件声明为private或这protect, 那么定义事件的类外部也不能对事件进行"+="和"-="操作

事件编程

其实,上面的例子只是一个简单的演示。在很多情况下,事件使用过程中都会结合两个参数:事件源和事件参数。

所以,在事件编程中,可以参考下面一些规范:

1.统一以eventnameeventhandler方式命名委托变量:eventname是事件的名称
2.delegate接受两个参数,参数名统一命名为sender和e:第一个参数类型是object,第二个参数是事件参数类型,以eventnameeventargs命名,并且需继承于system.eventargs类
3.如果在事件中不需要传递任何数据,也需要声明两个参数:第一个参数就是默认的object sender,第二个参数可以使用系统默认的system.eventargs类

总结

通过本文介绍了事件的概念以及原理,解释了通过事件如何封装委托实例,并解决委托例子中遇到的两个问题。

同时了解了事件的使用:

通过event关键字声明事件,

复制代码 代码如下:

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

事件的调用,由于事件本质上是委托类型,调用事件与调用委托一样,但是事件的调用只能发生在定义事件的类的内部

如您对本文有疑问或者有任何想说的,请点击进行留言回复,万千网友为您解惑!

相关文章:

验证码:
移动技术网