ajax 在目前 web 领域已广泛应用,其真正核心只不过是一个封装好了的 js 库。最五花八门的莫过于 asp.net 的控件,我个人认为 ajax 只是一个轻量级的东西,根本没有必要将它写成服务器组件,如 ajax.net、AjaxControlToolkit 等。所以我一直视这些组件为垃圾。。。
以下说明我为什么不认同 ajax 的相关组件:
1、将 ajax 封装得太死,出现问题难于调试;
2、组件过于臃肿,太多没有用的功能;
3、依赖性太强,少一样不可,如 webform;
传统的 ajax 应用到项目中,会出现由于项目的 ajax 应用太多,服务端的 ajax 响应函数难于管理,我们到底要把这些响应函数部署在什么位置?
假如
index.aspx 里有三个 ajax 响应函数,我们可以把这三个响应函数放在 index.aspx 中,并于 Page_Load 事件中根据参数不同来调用这些函数。
index.aspx 里还使用了 abc.ascx,这个控件里面也用到了 ajax,那么 abc.ascx 里的响应函数又应该放在哪里?index.aspx 中吗?如果 index2.aspx 中也用到了 abc.ascx,难道还要复制 index.aspx 的处理函数到 index2.aspx,这样肯定不行;
我们可以新建一个 ajax.aspx 来处理所有的 ajax 响应,php/asp 都可以这样做。这样管理还是不理想,
最理想的管理方法应该是页面、控件与响应函数集成。ICallBackEventHandler 可以做到,但是它依赖 webform,假如你的页面中没有一个 <form runat="server">,则会失效。
问题回归到如体使用轻量级 ajax,并实现页面/控件与 ajax 响应函数集成。
Reflector 查看 System.Web.UI.Page 里面处理 ICallBackEventHandler 模块后,我们可以这样做:
1、所有页面继承 BasePage;
2、BasePage 继承 System.Web.UI.Page,重写 BasePage.OnLoad;
3、实现 AjaxHandler
4、添加 Ajax 类,实现 Ajax.Register 静态方法
代码如下:

public class BasePage : Page
{

protected override void OnLoad(EventArgs e)
{
AjaxHandler();
base.OnLoad(e);
}


ajax handler#region ajax handler
//我喜欢用 json 格式数据来返回给客户端处理

protected void AjaxHandler()
{
string ajax_target = Request.Form["ajax_target"];
if (string.IsNullOrEmpty(ajax_target)) return;

Control target = FindControl(ajax_target);
if (target == null) return;

MethodInfo method = target.GetType().GetMethod("IAjax");
if (method == null) return;

Response.Cache.SetNoStore();

Response.Write("(");
method.Invoke(target, null);
Response.Write(")");
Response.End();
}
#endregion
}


public class Ajax
{

public static string Register(Control control, string function, string argument)
{
if (control == null) return "//This control can not be NULL.";
return string.Format("ajax({0},'{1}','ajax_target={2}{3},'post','{4}');", function, HttpContext.Current.Request.Url.PathAndQuery,
HttpContext.Current.Server.UrlEncode(control.UniqueID), string.IsNullOrEmpty(argument) ? "'" : "&'+" + argument,
HttpContext.Current.Response.Charset);
}
}
我在 Page.OnLoad 前处理 ajax,通过 Request.Form["ajax_target"] 得到将要调用的目标,然后反射到目标的 IAjax 这个名字的函数(这里命名有问题,我随便命的),最后通过反射调用它并 Response.End
Ajax.Register 是注册一个 Ajax,和 ICallBackEventHandler 的使用方法一样,但不同的是现在不需要依赖 webform 和 asp.net 自带的 js库。它依赖传统的 ajax.js 库,并且通用 POST 自己传入参数
我没有写 ClientScript.RegisterClientScriptBlock 页面自动包含 ajax.js,有兴趣的可以自己改,到时你会遇到 ClientScript.RegisterClientScriptBlock 也依赖 webform,哈哈。。。你需要重新实现相应方法才可以,有兴趣可以直接询问我。
使用方法:abc.ascx
<%@ Control Language="C#" ClassName="abc" %>
<script runat="server">
public void IAjax() {
int country_id;
int.TryParse(Request.Form["country_id"], out country_id);
Response.Write("[");
List<CityInfo> citys = City.GetItemsByCountry_id(country_id);
foreach (CityInfo city in citys) {
Response.Write(string.Format("['{0}',{1},'{2}'],", city.Code, city.Id, city.Name); //这里我没有处理 js 问题,你自己处理
}
Response.Write("]");
}
</script>
<input type="text" id="country_id">
<input type="button" value="GO" onclick="getCitys()">
<script type="text/javascript">
function getCitys() {
var receive = function(rt) {
alert(rt);
};
var args = 'country_id=' + $('country_id').value;
<%= Ajax.Register(this, "receive", "args") %>
}
</script>
这样一来,我们就轻松的将 ajax 部署到 UserControl/Page 中,不再依赖 asp.net 的不太好的东西。
ajax.js 代码:

Code
var ajaxPool = [];
function ajax(delegate, uri, data, method, encoding, errors) {
if (!uri) { alert('错误:未指知异步请求URI,操作失败。'); return; }
if (!data) data = '';
if (!method) method = 'get';
if (!encoding) encoding = 'utf-8';
var http = ajaxPool.pop();
if (typeof http == 'undefined' || http == null) {
var flag = 0;
if (window.XMLHttpRequest) {
http = new XMLHttpRequest();
} else if (window.ActiveXObject) {
var progIDs = [ 'Msxml2.XMLHTTP', 'Microsoft.XMLHTTP' ];
for (var i = 0; i < progIDs.length; i++) {
try {
http = new ActiveXObject(progIDs[i]);
break;
} catch (ex) { }
}
} else {
flag = 1;
}
if (flag) {
var div = document.createElement('div');
var rnd = hexRnd('FFFFFF', 16);
div.innerHTML = '<iframe id="nic_ajax_' + rnd + '" style="width:0px;height:0px;"></iframe>';
document.body.appendChild(div);
http = document.getElementById('nic_ajax_' + rnd);
http.onreadystatechange = function() {
this.status = this.contentWindow.document.body.innerHTML == '' ? 0 : 200;
this.responseText = this.contentWindow.document.body.innerHTML;
statechange();
}
}
}
if (http.tagName && http.tagName.toLowerCase() == 'iframe')
http.src = uri;
else
{
try {
http.abort();
http.open(method, uri, true);
if (method == 'post')
http.setRequestHeader('content-type', 'application/x-www-form-urlencoded');
http.setRequestHeader('content-type', 'text/html;charset=' + encoding);
http.setRequestHeader('user-agent', 'MSIE 6.0');
http.onreadystatechange = statechange;
http.send(data);
} catch(e) {
alert('错误:(在准备申请“' + uri + '”资源时)\r\n' + e.message);
ajaxPool.push(http);
}
}
function hexRnd(hex) {
return parseInt(Math.random() * parseInt(hex, 16)).toString(16);
}
function statechange() {
if (http.readyState == 4) {
switch (http.status) {
case 200:
if (delegate) delegate(http.responseText, http);
ajaxPool.push(http);
break;
default:
ajaxPool.push(http);
if (!errors) errors = 0;
if (++errors < 100)
setTimeout(function() {
ajax(delegate, uri, data, method, encoding);
}, 1000);
window.defaultStatus = '错误:(在申请“' + uri + '”资源时)\r\n网络断线或超时(' + errors + '次。';
break;
}
}
}
}