在asp.net中客户端与服务器端的交互默认都是整页面提交, 此时客户端将当前页面表单中的数据(包括一些自动生成的隐藏域)都提交到服务器端,服务器从新实例化一个当前页面类的实例响应这个请求,而后将整个页面的 内容从新发送到客户端,这种处理方式对运行结果没什么影响,不过这种方式加剧了网络的数据传输负担、加大了服务器的工做压力,而且用户还须要等待最终处理 结果。假如是咱们但愿有这么一个功能,当用户填写完用户名以后就检查服务器数据库里是否已存在该用户名,若是存在就给出已经存在此用户名的提示,若是不存 在就提示用户此用户名可用,对于这种状况其实只须要传递一个用户名做为参数便可,上面的作法却须要提交整个表单,有点小题大作。解决上面的问题的办法目前 主流作法有三种:纯javascript实现、微软Ajax类库实现还有用AjaxPro实现。后两种作法在稍后的文章中会讲到,这里我讲另一种实现: 经过回调技术。javascript
建立实现回调技术的网页与普通asp.net网页相似,只不过还须要作如下特殊工做:
(1)让当前页面实现 ICallbackEventHandler接口,这个接口定义了两个方法:string GetCallbackResult ()方法和void RaiseCallbackEvent (string eventArgument)方法。其中GetCallbackResult ()方法的做用是返回以控件为目标的回调事件的结果,RaiseCallbackEvent()方法的做用是处理以控件为目标的回调事件。
(2) 为当前页提供三个javascript客户端脚本函数。一个javascript函数用于执行对服务器的实际请求,在这个函数中能够提供一个字符串类型的 参数发送到服务器端;另外一个javascript函数用于接收服务器端方法的执行后返回的字符串类型结果,并处理这个结果;还有一个是执行对服务器请求的 帮助函数,在服务器代码中经过GetCallbackEventReference()方法获取这个方法的引用时由asp.net自动生成这个函数。
下面我以一个详细的例子来说述如何使用回调,用Dreamweaver建立一个Register. aspx页面,代码以下:html
1 <%@ Page Language="C#" %>
2 <%@ Implements Interface="System.Web.UI.ICallbackEventHandler" %>
3 <%@ Import Namespace="System.Text" %>
4 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN""http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
5
6 <script language="javascript">
7 //客户端执行的方法
8
9 //下面的方法是接收并处理服务器方法执行的返回结果
10 function Success(args, context)
11 {
12 message.innerText = args;
13 }
14 //下面的方式是当接收服务器方法处理的结果发生异常时调用的方法
15 function Error(args, context)
16 {
17 message.innerText = '发生了异常';
18 }
19 </script>
20
21 <script language="c#" runat="server">
22 string result="";
23 // 定义在服务器端运行的回调方法.
24 public void RaiseCallbackEvent(String eventArgument)
25 {
26
27 if (eventArgument.ToLower().IndexOf("admin") != -1)
28 {
29 result = eventArgument + "不能做为用户名注册。";
30 }
31 else
32 {
33 result = eventArgument + "能够注册。";
34 }
35
36 //throw new Exception();
37 }
38
39 //定义返回回调方法执行结果的方法
40 public string GetCallbackResult()
41 {
42 return result;
43 }
44
45 //服务器上执行的方法
46 public void Page_Load(Object sender, EventArgs e)
47 {
48 // 获取当前页的ClientScriptManager的引用
49 ClientScriptManager csm = Page.ClientScript;
50
51 // 获取回调引用。会在客户端生成WebForm_DoCallback方法,调用它来达到异步调用。这个方式是微软写的方法,会被发送到客户端
52 //注意这里的"Success"和"Error"两个字符串分别客户端代码中定义的两个javascript函数
53 //下面的方法最后一个参数的意义:true表示执行异步回调,false表示执行同步回调
54 String reference = csm.GetCallbackEventReference(this, "args", "Success", "", "Error",false);
55 String callbackScript = "function CallServerMethod(args, context) {\n" +reference + ";\n }";
56
57 // 向当前页面注册javascript脚本代码
58 csm.RegisterClientScriptBlock(this.GetType(), "CallServerMethod",
59 callbackScript, true);
60 }
61 </script>
62
63 <html >
64 <head runat="server">
65 <title>無題のページ</title>
66 </head>
67 <body>
68 <form id="form1" runat="server">
69 <table border="1" cellpadding="0" cellspacing="0" width="400px">
70 <tr>
71 <td width="100px">用户名</td><td><input type="text" size="10" maxlength="20"id="txtUserName" onblur="CallServerMethod(txtUserName.value,null)" /><span id="message"></span></td>
72 </tr>
73 <tr>
74 <td>密码</td><td><input type="password" size="10" maxlength="20" id="txtPwd" /></td>
75 </tr>
76 </table>
77 </form>
78 </body>
79 </html>java
上面的页面中我已经添加了足够详尽的注视,不过我仍是要说明几点:
(1)web
这句表示当前页面实现了ICallbackEventHandler接口,若是采用页面与代码分离的模式,后台cs代码则应是:数据库
(2)c#
这里有一个onblur="CallServerMethod(txtUserName.value,null),表示当用户名文本框失去焦点以后激发CallServerMethod这个客户端方法,这个客户端方法是由asp.net动态生成的。
(3)浏览器
中的"Success"和"Error"分别表明客户端的javascript函数,能够在代码中见到,其中"Success"表明调用服务器端方法成功后要执行的客户端方法名,"Error"表明调用服务器端方法失败时调用的客户端方法名。服务器
该页面在客户端生成的HTML代码以下:网络
1
2 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
3
4 <script language="javascript">
5 //客户端执行的方法
6
7 //下面的方法是接收并处理服务器方法执行的返回结果
8 function Success(args, context)
9 {
10 message.innerText = args;
11 }
12 //下面的方式是当接收服务器方法处理的结果发生异常时调用的方法
13 function Error(args, context)
14 {
15 message.innerText = '发生了异常';
16 }
17 </script>
18
19
20
21 <html xmlns="http://www.w3.org/1999/xhtml" >
22 <head><title>
23 無題のページ
24 </title></head>
25 <body>
26 <form name="form1" method="post" action="Register.aspx" id="form1">
27 <div>
28 <input type="hidden" name="__EVENTTARGET" id="__EVENTTARGET" value="" />
29 <input type="hidden" name="__EVENTARGUMENT" id="__EVENTARGUMENT" value=""/>
30 <input type="hidden" name="__VIEWSTATE" id="__VIEWSTATE"value="/wEPDwUKMTUxMzcyMjQyN2RktUwTa0pYHOlQ0OTLFd6fte0EGow=" />
31 </div>
32
33 <script type="text/javascript">
34 <!--
35 var theForm = document.forms['form1'];
36 if (!theForm) {
37 theForm = document.form1;
38 }
39 function __doPostBack(eventTarget, eventArgument) {
40 if (!theForm.onsubmit || (theForm.onsubmit() != false)) {
41 theForm.__EVENTTARGET.value = eventTarget;
42 theForm.__EVENTARGUMENT.value = eventArgument;
43 theForm.submit();
44 }
45 }
46 // -->
47 </script>
48
49
50 <script src="/web/WebResource.axd?d=jAi7Db33LHl_8HdPSGuzAg2&t=633608119083845334" type="text/javascript"></script>
51
52
53 <script type="text/javascript">
54 <!--
55 function CallServerMethod(args, context) {
56 WebForm_DoCallback('__Page',args,Success,"",Error,false);
57 }// -->
58 </script>
59
60 <table border="1" cellpadding="0" cellspacing="0" width="400px">
61 <tr>
62 <td width="100px">用户名</td><td><input type="text" size="10" maxlength="20"id="txtUserName" onblur="CallServerMethod(txtUserName.value,null)" /><span id="message"></span></td>
63 </tr>
64 <tr>
65 <td>密码</td><td><input type="password" size="10" maxlength="20" id="txtPwd" /></td>
66 </tr>
67 </table>
68
69
70 <script type="text/javascript">
71 <!--
72
73 WebForm_InitCallback();// -->
74 </script>
75 </form>
76 </body>
77 </html>
78asp.net
在生成的HTML代码中多了几段javascipt教本块,下面分别说明:
(1)第一部分
这部分代码是每一个asp.net页面发送到客户端都会生成的,用于提交当前表单,其中eventTarget参数表示激发提交事件的控件,eventArgument参数表示发生该事件时的参数信息。
(2)第二部分
这部分代码是用来生成一些用于Ajax调用的js脚本。说穿了,asp.net之因此开发起来方便,是由于微软在幕后默默地为咱们作了不少工做,回调的本质其实就是Ajax调用。
咱们能够将“/WebResource.axd?d=CcZ-_AaHZnD65xnNHEUijg2&t=633578466781093750”这部分拷贝到浏览器地址栏中,以下图:
回车以后会弹出一个下载文件对话框,以下图:
将这个页面保存到本地,虽然默认的保存文件的后缀为“.axd”,但它实际上是一个文本文件,里面是一些javascript代码,咱们能够用记事本打开,在里面咱们能够看到“WebForm_DoCallback”这个方法,以下:
在这个axd文件里作了不少幕后工做,因此咱们的回调才相对比较简单。
(3)第三部分
这部分代码是后台生成的,经过获取Page类的ClientScript属性,也就是ClientScriptManager的实例注册到页面的, 里面定义了两个javascript函数:CallServerMethod函数和WebForm_DoCallback函数,而且是在 CallServerMethod函数中调用WebForm_DoCallback函数。
(4)第四部分
这部分代码也是幕后生成的,这个javascript函数也能够在那个axd文件中找到。以下图:
后,会获得能够注册的提示,以下图:
当咱们输入“admin”做为用户名时的结果:
另外,咱们将服务器端执行的方法作以下处理,也就是RaiseCallbackEvent(String eventArgument)这个方法,咱们在这里抛出一个异常,代码以下:
再次运行,不管咱们以什么做为用户名,都会获得以下结果:
之因此会出现“发生了异常”这个字符串,是由于咱们定义了function Error(args, context)这个javascript函数,而且把它做为调用服务器端方法发生异常时的客户端处理函数,它的处理方式就是显示“发生了异常”这个字符串。