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页面中,运行测试下。
单击按钮后,代码的执行并没有进入断点处函数。为什么?
因为IPostBackEvnetHander接口只提供捕获回发事件,但并没有回发就谈不上捕获了(客户端button类型的input按钮不会触发回发事件的)。将input类型该为submit类型再试下,这次可以了吧。
(5).控件自动回发实现
上面只用当按钮类型改为submit类型时才能捕获回发事件,但总不能任何的控件都需要带上一个提交按钮吧,别急,下面就说说控件的客户端自动回发。
要实现控件客户端的自动回发,可以采用两种方式——通过GetPostBackEventReference或GetPostBackClientHyperlink方法获取客户端回发函数的引用。
修正上面的代码如下:
protected override void RenderContents(HtmlTextWriter output) { //output.Write(Text); //output.Write("<input type='button' name=\"{0}\" value='[点击我]' ></input>",// this.UniqueID); output.Write("<input type='button' name=\"{0}\" value='[点击我]' onclick=\"{1}\"></input>", this.UniqueID, Page.ClientScript.GetPostBackEventReference(this, "")); }
或者
protected override void RenderContents(HtmlTextWriter output) { output.Write("<input type='button' name=\"{0}\" value='[点击我]' onclick=\"{1}\"></input>" ,this.UniqueID,this.Page.ClientScript.GetPostBackEventReference(this,"")); }
运行程序,测试下,你的控件现在就可以自动回发了。
(6).双击Default.asp页面中的【点击我】按钮,Default.asp.cs文件中为自动生成该按钮的单击事件函数。
protected void ServerControl1_Click(object sender, EventArgs e){ }
打上断点,单步跟下,在该函数中可以调用页面类中的方法和属性了。这就是Button对象通过事件调用Page对象中的方法和属性了。
5.异步回调的实现
异步回调的处理就不是控件封装中的内容了,单为了让大家能够分清回发和回调的区别,还是讲解下。
异步回调是在页面类中处理的,同样,页面类需要继承ICallBackEventHandle接口并实现接口方法来增加该页面类的异步回调的功能。这里,就需要从JavaScript开始着手了,还是沿用上面的工程进行实例讲解。
(1).在上面的Default.asp页面中添加一个按钮和用于显示结果的Span元素。
<html xmlns="http://www.w3.org/1999/xhtml" > <head runat="server"> <title>无标题页</title> </head> <body> <form id="form1" runat="server"> <div> <cc1:ServerControl1 ID="ServerControl1" runat="server" onclick="ServerControl1_Click" /> <br /><br /> <input id="ButtonCallBack" type="button" value="[实现了异步回调功能的Button]" /> <br /> <div>显示异步回调两个参数结果:<br /> Args1. <span id="ResultShowArg1"></span><br /> Args2. <span id="ResultShowArg2"></span> </div> </div> </form> </body> </html>
(2).Default.asp页面后台文件类继承ICallBackEventHandle接口并实现接口方法。
public partial class _Default : System.Web.UI.Page, ICallbackEventHandler { protected void Page_Load(object sender, EventArgs e){ } protected void ServerControl1_Click(object sender, EventArgs e){ } #region ICallbackEventHandler 成员 public string GetCallbackResult() { throw new NotImplementedException(); } public void RaiseCallbackEvent(string eventArgument) { throw new NotImplementedException(); } #endregion }
(3).在页面类的Page_Load事件函数中加入下面代码:
protected void Page_Load(object sender, EventArgs e) { string cbRe = Page.ClientScript.GetCallbackEventReference(this, "arg", "CallBackServe", "context"); string callbackscript; callbackscript = "function CallPageServe(arg,context)" + "{" + cbRe + ";}"; Page.ClientScript.RegisterClientScriptBlock(this.GetType(), "CallPageServe", callbackscript, true); }
其中首先通过page对象引用ClientScript.GetCallbackEventReference()方法获取该页面的回调函数的应用,然后将该应用通过RegisterClientScriptBlock()方法注册到客户端页面中。
这里需要说明下,其中GetCallbackEventReference方法的第一个参数是控件对象的引用,通常情况下是this;第二个和第四个参数(“arg”和“context”)是处理客户端想服务端回调的函数CallPageServe函数的传入参数;真正重要的就是第三个参数“CallBackServe”,这是回调过程中服务端处理完后返回到客户端后处理返回数据的JS函数名。
(4).返回到Default.asp页面,双击“ButtonCallBack”按钮,为该按钮注册客户端JS单击事件函数ButtonCallBack_onclick()。
<script language="javascript" type="text/javascript"> function ButtonCallBack_onclick() { CallPageServe("args1", "args2"); } function CallBackServe(returnValue, context) { document.getElementById("ResultShowArg1").innerHTML = returnValue; document.getElementById("ResultShowArg2").innerHTML = context; } </script>
其中ButtonCallBack_onclick()中调用的CallPageSever()就是回调的触发函数。而CallBackServe()就是客户端处理回调返回数据的JS函数。
(5).回到Default.asp.cs文件,为该页面类添加一个String 字符串对象,并在ICallBackEventHandle接口方法的函数体内做相应的处。
string returnValue = ""; #region ICallbackEventHandler 成员 public string GetCallbackResult(){ //throw new NotImplementedException(); return returnValue; } public void RaiseCallbackEvent(string eventArgument) { //throw new NotImplementedException(); returnValue = eventArgument; } #endregion
(6).运行检验下,单击ButtonCallBack按钮后显示回调的结果。
在你感兴趣的地方打上断点单步跟下吧,你会发现还有其他有趣的东东。
还是附上全部源码吧。
Default.asp.cs文件
using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.UI; using System.Web.UI.WebControls; using PostBackEventControlDemo; namespace test { public partial class _Default : System.Web.UI.Page,ICallbackEventHandler { protected void Page_Load(object sender, EventArgs e) { string cbRe = Page.ClientScript .GetCallbackEventReference(this, "arg", "CallBackServe", ""); string callbackscript; callbackscript = "function CallPageServe(arg,context)" + "{" + cbRe + ";}"; Page.ClientScript.RegisterClientScriptBlock(this.GetType(), "CallPageServe", callbackscript, true); } protected void onClickHandle(object sender, EventArgs e) { String txt = ServerControl11.Text; int i = 0; } #region ICallbackEventHandler Members string returnValue = ""; public string GetCallbackResult() { return returnValue; //throw new NotImplementedException(); } public void RaiseCallbackEvent(string eventArgument) { returnValue = eventArgument; //throw new NotImplementedException(); } #endregion } }
Default.asp
<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="Default.aspx.cs" Inherits="test._Default" %> <%@ Register assembly="PostBackEventControlDemo" namespace="PostBackEventControlDemo" tagprefix="cc2" %> <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml" > <head runat="server"> <title></title> </head> <body> <form id="form1" runat="server"> <div> <cc2:ServerControl1 ID="ServerControl11" runat="server" OnClick="onClickHandle"/> <br /> <br /> <input id="ButtonCallBack" type="button" value="[实现了异步回调功能的Button]" onclick="return ButtonCallBack_onclick()"/> <div>显示异步回调两个参数结果:<br /> ARG1 <span id="ResultShowArg1" ></span><br /> ARG2 <span id="ResultShowArg222" ></span> </div> <script language="javascript" type="text/javascript"> function ButtonCallBack_onclick() { CallPageServe("args1|args2"); } function CallBackServe(returnValue) { var str = returnValue.split('|'); document.getElementById("ResultShowArg222").innerHTML = str[1]; document.getElementById("ResultShowArg1").innerHTML = str[0]; } </script> </div> </form> </body> </html>
ServerControls1
using System; using System.Collections.Generic; using System.ComponentModel; using System.Linq; using System.Text; using System.Web; using System.Web.UI; using System.Web.UI.WebControls; namespace PostBackEventControlDemo { [DefaultProperty("Text")] [ToolboxData("<{0}:ServerControl1 runat=server></{0}:ServerControl1>")] public class ServerControl1 : WebControl,IPostBackEventHandler { [Bindable(true)] [Category("Appearance")] [DefaultValue("")] [Localizable(true)] public string Text { get { String s = (String)ViewState["Text"]; return ((s == null) ? "[" + this.ID + "]" : s); } set { ViewState["Text"] = value; } } protected override void RenderContents(HtmlTextWriter output) { output.Write("<input type='button' name=\"{0}\" value='[点击我]' onclick=\"{1}\"></input>" ,this.UniqueID,this.Page.ClientScript.GetPostBackEventReference(this,"")); } #region IPostBackEventHandler Members public void RaisePostBackEvent(string eventArgument) { OnClick(EventArgs.Empty); } #endregion public event EventHandler Click; protected virtual void OnClick(EventArgs e) { if(Click != null) { Click(this,e); } } } }