一切福田,不離方寸,從心而覓,感無不通。

Asp.net控件开发学习笔记(八)—-服务器控件事件

 

事件和委托之间的暧昧关系往往是大多Web Developer在学习.net中的一个里程碑,当明白事件和委托的关系后,.net水平往往就上了一个新的台阶.下面说到服务器控件的事件模型.

在任何一个服务器编程开发框架中,事件都是解耦功能和具体实现的一剂良方,Asp.net当然也不例外。比如说吧,页面上的buttonclick事件表示它的功能,而具体的实现将会被分离交给Developer来进行具体实现。

传统的编程模型和基于事件的编程模型可以用下图进行简约概括:



我们可以看出事件极大的简化了编程工作,客户端程序只需要注册到事件并且和事件的签名保持一致(即参数个数和类型相同)即可。在事件发生后,客户端程序会被通知并执行相应实现(.net framework的事件正是观察者模式的最好例子:-)

 

Asp.net通过ViewStateHttp Post协议巧妙的实现了让开发人员感觉貌似控件能像WinForm程序中那样记住自己的状态。这使Asp.net可以在不使用客户端javascript的情况下,而实现数据回传。

 



    上面的图例展示出
TextBox通过暴露相应的事件来通知被注册的函数.还记得前面所说的IPostDataHandler接口嘛,大多数服务器控件的事件都是通过ViewState来将数据以Http Post协议传回服务器,服务器根据回传数据的不同来引发相应事件。因此我们可以看出Button控件生成的<input type=”submit” />是有往服务器提交的功能的,而其他控件比如DropDownList或者是Checkbox是没有像服务器提交的功能。因此引发服务器事件便无从谈起。Asp.net通过在客户端设置javascript事件来引发向服务器的Http Post回传。而这一切仅仅需要将AutoPostBack属性设置为true.

 

 

.net FrameWork 事件模型

.net framework提供了基于委托的使不同类之间进行异步交互的机制。下面先简单说一下事件的核心——-委托

 

Delegate

委托在一定程度上有点像接口,是在发布者和订阅者之间的一个协议。接口是制订类的数据成员以及成员函数的签名。而委托,是制订单个函数的签名.

创建一个委托的实例是通过创建一个和委托相匹配的函数。

MSDN里把委托比喻成类型安全的函数指针。但是委托不仅限于此,因为.net framework大大的扩展了这个“类型安全的指针”,在CLR via C#这本书里说委托的实质上是一个类。因此委托可以按照次序依次调用多个匹配的方法,无论是静态方法还是实例方法。

委托有两个部分,委托的声明和委托实例的声明。

委托的声明代码会像:

publicdelegatevoidfoo(string A);

而委托实例的声明会像

foo f=new foo(MethodName)

 

下面通过一个简单的Demo说明一下:

 

Demo 委托

先写一个简单的类:

publicclassDelegateDemo

    {

publicdelegatevoidDeleDemo(string a);

publicstaticvoid FunctionA(string a)

        {

HttpContext.Current.Response.Write("静态方法,传入的参数是:" + a + "<br />");

        }

publicvoid FunctionB(string a)

        {

HttpContext.Current.Response.Write("实例方法,传入的参数是:" + a + "<br />");

        }

}

下面是客户端代码:

DelegateDemo dd = newDelegateDemo();

DelegateDemo.DeleDemo d = newDelegateDemo.DeleDemo(dd.FunctionB);

        d += DelegateDemo.FunctionA;

    d("参数");

 

最后的输出结果为:

实例方法,传入的参数是:参数
静态方法,传入的参数是:参数

 

通过上面小Demo可以发现委托不仅仅是“类型安全的指针”,并且委托是按照次序调用实例方法和静态方法.

 

事件

C#里有专门用于声明事件的event关键字,一个典型的事件生命会像:

publiceventEventHandler click;

 

事件关键字后面是委托,再后面是事件名称,命名事件的名称最好是动词,表名某些事情发生了。比如click,Init,Load,TextChanged这样。由此可以看出事件其实就是特殊的委托。因为所有的事件都是继承于System.EventHandler.

 

EventHandler 委托

所有asp.net内置控件事件处理函数的签名都和EventHandler或者继承于它的子类保持一致。它的原型是:

[SerializableAttribute]

[ComVisibleAttribute(true)]

publicdelegatevoidEventHandler(

Object sender,

EventArgs e

)

 

第一个参数表示引发事件的对象,第二个参数表示引发事件后所要传给处理程序的参数。

 

在一般情况下,开发人员最好是按照这种签名格式来声明函数事件的委托.

 

在控件内部声明事件后,你必须在需要的情况下引发事件,直接引发事件是非常不好的做法。而在asp.net预定义的控件中都使用了如下方法:

声明一个virtual protected void的方法,命名方式为On+事件名称.下面是一个例子:

protectedvirtualvoid OnClick(EventArgs e)

        {

if (Click != null)

                Click(this, e);

       }

这个方法首先做的是先检查客户端方法是否注册,如果有客户端方法进行了注册,则引发事件。

http://www.cnblogs.com/CareySon/archive/2009/10/12/1581792.html