日期:2014-05-17  浏览次数:20950 次

ASP.NET服务器控件封装-【事件】-1.1【事件回发.异步回调】

1.事件以及为什么需要事件驱动机制

在C#语言详解一书中对事件的定义是“事件是一种使对象或类能够提供通知的成员”,在这里换句话说就是页面中已注册事件的对象能够对用户的操作进行捕获并处理。那么为什么需要引用事件机制呢?

大家都知道,如果在类A的实例对象中创建了一个类B的实例对象,那么在类A的实例对象中就可以通过该类B的实例对象调用类B公开的任何方法和属性等。就像用户Page对象中包含了创建了一个TextBox对象,Page对象就可以通过TextBox对象去调用Text属性。但是如果需要在上述的TextBox对象中调用Page对象中的某些属性或方法又该怎样处理呢?显然包含调用就行不通了,事件机制正好解决该问题。

现就TextBox的TextChanged事件来描述下。

首先需要在TextBox中声明TextChanged委托,并通过页面注册将该委托和页面类处理事件函数关联起来

<asp:TextBox ID="TextBox1" runat="server"  OnTextChanged="TextBox1_TextChanged"></asp:TextBox>

?

(上述代码就是将TextBox1的TextChanged事件注册到页面类中)。

然后在TextBox类体内调用TextChanged委托关联的事件函数就可以了。这样,当页面类对象中TextBox控件的Text发生改变时TextBox首先执行完自己内部逻辑处理后,就会调用页面类中OnTextChanged事件函数TextBox1_TextChanged(),而TextBox1_TextChanged()函数是页面类中的一个方法,所以TextBox就可以在该函数内调用页面类中的其他方法和属性了。(当然也是可以调用TextBox1自己的)。

?

2.回发的原理

Web开发的人员都知道,客户端回发到服务端的事件只有一个,那么在服务端怎样的区分用户执行了怎样的操作呢?这里首先从ASP.NET的页面请求说起了。

在ASP.NET中处理页面时,前后两个页面之间是无状态连接的,也就是说客户端的前后两次请求是相互独立的,服务端不会保存前一次请求的页面状态。如此就引入了视图状态机制(我会在以后的文章中和大家分享下ASP.NET开发的视图状态和控件状态机制的研究乐趣)来处理前后两次请求的逻辑处理,其原理就是在前一次请求发生后服务端将页面的逻辑信息保存在一个隐藏的字段中回送到客户端,当后一次请求发生时服务端首先取出该隐藏字段中的值并恢复到各个视图控件中,等逻辑处理完后再将新的数据保存到该隐藏字段会送到客户端,从而延续了两次页面之间的状态信息。

本文中要说明的回发就是在视图状态机制基础上完成的,也就是通过比较发送到服务端的控件当前值和保存在隐藏字段中的旧值,从而决定是否触发哪些事件。当然,自定义的控件类必须通过继承IPostBackDataHander接口来完成事件的回发功能。

?

3.异步回调的原理

和回发不同,回调就是从客户端到服务端,在服务端处理相关逻辑完了将处理的数据返回给客户端,就相当于客户端调用了服务端的方法(这和Web Service很像)一样,将处理的数据作为返回的结果供客户端程序处理。

异步回调过程中,ASP.NET会修改页面正常的生命周期来处理请求,而回发则是ASP.NET会用一个完整的页面生命周期处理请求。

另外,异步回调过程中,用户还可以进行其他的操作,客户端页面并不会重新刷新。

?

4.事件回发的实现

上面说过,客户端到服务端的事件只有一个——回发事件。那么,服务端怎样使自己的控件捕获该回发事件呢?

你需要将你的控件类继承IPostBackEvnetHander接口并实现该接口中唯一的一个方法:RaisePostBackEvent(string eventArgument)。

下面就以一个简单的实例说明下:

?

(1).首先创建一个服务器控件项目和测试项目。(这里介绍了项目的创建方法,以后就不再说明了哦!)

打开VS2008新建项目,在弹出的对话框中选中ASP.NET服务器控件模板,输入你项目名并选择路径,如下图1。

图1

然后在资源管理器中的“解决方案...”上右键“添加”=>“新建项目”(如图2所示),弹出如图1对话框,选中“ASP.NET Web应用程序”后输入项目名(这里仅只为了测试控件的一些处理,所以命名为test)确定。

(2).继承IPostBackEvnetHander接口并实现接口方法。

如下图所示,ServerControl1继承了IPostBackEvnetHander接口,将鼠标悬停在该接口上片刻会弹出实现接口方法提示对话框,如下图3所示。

选择“实现接口...”就可自动生成该接口需要实现的方法的空函数代码。

#region IPostBackEventHandler 成员        
public void RaisePostBackEvent(string eventArgument)        
{           
         throw new NotImplementedException();       
 } 
#endregion

?

(3).接下来,就开始实现该方法了。

protected override void RenderContents(HtmlTextWriter output)        {            
//output.Write(Text);            
output.Write("<input type='button' name=\"{0}\" value='[点击我]' ></input>",  this.UniqueID);                    
}                
#region IPostBackEventHandler 成员        
public void RaisePostBackEvent(string eventArgument)       
{            
//throw new NotImplementedException();            
OnCliclk(EventArgs.Empty);        
}        
#endregion        
/* Click事件委托 */        
public event EventHandler Click;        
protected virtual void OnCliclk(EventArgs e)        
{           
                 if (Click != null)            
                {                
                     Click(this, e);           
                 }        
}

?

上述代码中首先定义了一个Click委托和Onclick事件函数,用于与页面注册的单击事件函数关联。而在RaisePostBackEvent()方法中只调用了OnCliclk(EventArgs.Empty);方法,处理用户的逻辑。

?

(4).在控件的RaisePostBackEvent()方法处打下断点,并从工具栏中将上述控件(ServerControl1)拖放到test项目中的Default.asp页面中,运行测试下。

单击按钮