javascript 构建一个xmlhttp对象池合理创建和使用xmlhttp对象


Posted in Javascript onJanuary 15, 2010

如果我们在客户端频繁使用ajax技术,那么我们就不得不多次创建xmlhttp对象。当然,如您所知,我们可以改进创建的方式,比如使用全局变量来缓存一个实例(客户端的单例模式?!),对于同步方式的通信,这是很有效的,但是这样的方式对于异步通信会出现问题,因为没有了进程的堵塞,用户可能在上一次通信未完成时再次调用同一个xmlhttp实例,这样不等前一个调用的回调函数触发,前一次调用就被“覆盖”掉了(也就代表前一次调用失败)。建立一个保持xmlhttp实例的池,好处显而易见,最明显的优点就是我们不会创建冗余对象,同时也不会出现在同一个正在被调用的xmlhttp实例上出现再次被操作的情况。

具体实现思路:
我们使用一个数组来存储已创建的xmlhttp对象实例,然后每次调用从池中去取一个实例。xmlhttp实例通讯完毕后我们不用做任何处置,因为它自身的readyState属性可以标识出它是否可用,如果当时没有空闲的xmlhttp实例,且池中的实例数小于最大实例个数,那么就创建一个新的实例并放入池中。重新改进的实现代码如下:

//封装XMLHTTP的MyAjaxObj类 
var MyAjaxObj = new Object(); 
var maxXmlHttpCount = 5; //最多5个xmlhttp对象存在 MyAjaxObj.reqList = []; //可以清空里面的项 
MyAjaxObj.getFreeObj = function() { 
var req = null; 
var len = this.reqList.length; 
//先从当前的池里取 
for (var i = 0; i < len; i++) { 
if (this.reqList[i]) { 
if (this.reqList[i].readyState == 4 || this.reqList[i].readyState == 0) { 
req = this.reqList[i]; 
break; 
} 
} 
} 
//如果没有闲置的对象,自己独立创建 
if (req == null) { 
if (this.reqList.length < maxXmlHttpCount) { 
req = getXmlHttp(); 
this.reqList.push(req); 
} 
} 
return req; 
} 

//创建一个XMLHTTP对象,兼容不同的浏览器 
function getXmlHttp() { 
var xmlHttp = false; 
var arrSignatures = ["MSXML2.XMLHTTP.5.0", "MSXML2.XMLHTTP.4.0", 
"MSXML2.XMLHTTP.3.0", "MSXML2.XMLHTTP", 
"Microsoft.XMLHTTP"]; 
for (var i = 0; i < arrSignatures.length; i++) { 
try { 
xmlHttp = new ActiveXObject(arrSignatures[i]); 
return xmlHttp; 
} 
catch (oError) { 
xmlHttp = false; //ignore 
} 
} 
// throw new Error("MSXML is not installed on your system."); 
if (!xmlHttp && typeof XMLHttpRequest != 'undefined') { 
xmlHttp = new XMLHttpRequest(); 
} 
return xmlHttp; 
} 
/*封装XMLHTTP向服务器发送请求的操作 
url:向服务器请求的路径;method:请求的方法,即是get/post;***callback:当服务器成功返回结果时,调用的函数(类似c#回调函数)*** 
data:向服务器请求时附带的数据;urlencoded:url是否编码;cached:是否使用缓存; callBackError;当服务器返回错误时调用的函数 
*/ 
MyAjaxObj.send = function(url, method, callback, data, urlencoded, cached, callBackError) { 
var req = this.getFreeObj(); //从池里或者直接实例化一个XMLHTTP的实例 
//当XMLHTTP的请求状态发生改变时调用 (核心处理函数) 
req.onreadystatechange = function() { 
// 当请求已经加载 
if (req.readyState == 4) { 
// 当请求返回成功 
if (req.status == 200) { //或者 req.status < 400 
// 当定义了成功回调函数时,执行成功回调函数 
if (callback) 
callback(req, data); 
} 
// 当请求返回错误 
else { 
//当定义了失败回调函数时,执行失败回调函数 
if (callBackError) 
callBackError(req, data); 
} 
// 有池的管理,我们可以省却释放资源的方法 
// try { 
// delete req; 
// req = null; 
// } 
// catch (e) { 
// alert(e.message); 
// } 
} 
} 
//如果以POST方式回发服务器 
if (method.toUpperCase() == "POST") { 
req.open("POST", url, true); 
//请求是否需要缓存(只有在req.open之后才可以设置此项) 
if (cached) 
req.setRequestHeader("If-Modified-Since", "0"); 
//请求需要编码 
if (urlencoded) 
req.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded'); 
req.send(data); 
MyAjaxObj.reqList.push(req); 
} 
//以GET方式请求 
else { 
req.open("GET", url, true); 
//请求是否需要缓存 
if (cached) 
req.setRequestHeader("If-Modified-Since", "0"); 
req.send(null); 
MyAjaxObj.reqList.push(req); 
} 
return req; 
} 
//全部清除XMLHTTP数组元素,释放资源 
MyAjaxObj.clearReqList = function() { 
var len = MyAjaxObj.reqList.length; 
for (var i = 0; i < len; i++) { 
var req = MyAjaxObj.reqList[i]; 
if (req) { 
try { 
delete req; 
} catch (e) { } 
} 
} 
MyAjaxObj.reqList = []; 
} 
//进一步封装XMLHTTP以POST方式发送请求时的代码 
//isClear:是否清除XMLHTTP数组的所有元素;其他参数的意义见 MyAjaxObj.send 
MyAjaxObj.sendPost = function(url, data, callback, isClear, isCached, callBackError) { 
if (isClear) { 
MyAjaxObj.clearReqList(); 
} 
MyAjaxObj.send(url, "POST", callback, data, true, isCached, callBackError); //post方法需要编码 
} 
//进一步封装XMLHTTP以GET方式发送请求时的代码 
MyAjaxObj.sendGet = function(url, args, callback, isClear, isCached, callBackError) { 
if (isClear) 
MyAjaxObj.clearReqList(); 
return MyAjaxObj.send(url, "GET", callback, args, false, isCached, callBackError); 
}

最后再ps:上周周末和一个哥们聊天的时候谈到ajax应用中的xmlhttp对象。那哥们ms很“虔诚”地问我说xmlhttp怎么就异步通信了。我当时竟然毫不思索地说因为这个对象处理我们的请求调用是“异步”的(当然可以设置成同步的,不过这是一句废话),当前这个请求不会影响其他的操作。这个回答是很“官方”的,显然没有说到问题的本质。哥们,您的眼神儿有必要那么bs人么?现在稍作分析,个人认为其实每个xmlhttp异步请求都会触发一个回调函数,这个回调函数的调用不影响其他的操作,这个方法才是“异步”。如果对比c#里的异步处理回调方法,它们在原理上其实是相通的。 哈哈,现在终于想通了, 真是太骄傲,太有出息了,想到就兴奋!
Javascript 相关文章推荐
一个tab标签切换效果代码
Mar 27 Javascript
jquery 双色表格实现代码
Dec 08 Javascript
ASP.NET jQuery 实例2 (表单中使用回车在TextBox之间向下移动)
Jan 13 Javascript
中文输入法不触发onkeyup事件的解决办法
Jul 09 Javascript
JS 使用for循环遍历子节点查找元素
Sep 06 Javascript
jQuery实现复选框成对选择及对应取消的方法
Mar 03 Javascript
JavaScript计算两个日期时间段内日期的方法
Mar 16 Javascript
纯jQuery实现前端分页功能
Mar 23 jQuery
微信小程序中button组件的边框设置的实例详解
Sep 27 Javascript
vue组件数据传递、父子组件数据获取,slot,router路由功能示例
Mar 19 Javascript
JQuery使用属性addClass、removeClass和toggleClass实现增加和删除类操作示例
Nov 18 jQuery
利用js实现简易红绿灯
Oct 15 Javascript
extjs 为某个事件设置拦截器
Jan 15 #Javascript
利用onresize使得div可以随着屏幕大小而自适应的代码
Jan 15 #Javascript
javascript 不间断的图片滚动并可点击
Jan 15 #Javascript
Span元素的width属性无效果原因及解决方案
Jan 15 #Javascript
javascript实现的基于金山词霸网络翻译的代码
Jan 15 #Javascript
JQuery 引发两次$(document.ready)事件
Jan 15 #Javascript
用jQuery扩展自写的 UI导航
Jan 13 #Javascript
You might like
php注入实例
2006/10/09 PHP
解析CI的AJAX分页 另类实现方法
2013/06/27 PHP
php找出指定范围内回文数且平方根也是回文数的方法
2015/03/23 PHP
php从数据库查询结果生成树形列表的方法
2015/04/17 PHP
java script编程起步(第三课)
2007/01/10 Javascript
Jquery Ajax学习实例3 向WebService发出请求,调用方法返回数据
2010/03/16 Javascript
使用jquery mobile做幻灯播放效果实现步骤
2013/01/04 Javascript
javascript 获取iframe里页面中元素值的方法
2014/02/17 Javascript
node.js中的console.log方法使用说明
2014/12/09 Javascript
JS验证IP,子网掩码,网关和MAC的方法
2015/07/02 Javascript
JS组件Bootstrap实现下拉菜单效果代码
2016/04/26 Javascript
JavaScript 闭包机制详解及实例代码
2016/10/10 Javascript
vuejs开发组件分享之H5图片上传、压缩及拍照旋转的问题处理
2017/03/06 Javascript
JS实现给json数组动态赋值的方法示例
2020/03/19 Javascript
laravel5.4+vue+element简单搭建的示例代码
2017/08/29 Javascript
js构造函数创建对象是否加new问题
2018/01/22 Javascript
vue-router传参用法详解
2019/01/19 Javascript
Python与shell的3种交互方式介绍
2015/04/11 Python
python fabric实现远程部署
2017/01/05 Python
Python微信公众号开发平台
2018/01/25 Python
python数字图像处理实现直方图与均衡化
2018/05/04 Python
pandas进行数据的交集与并集方式的数据合并方法
2018/06/27 Python
Python爬虫逆向分析某云音乐加密参数的实例分析
2020/12/04 Python
html5配合css3实现带提示文字的输入框(摆脱js)
2013/03/08 HTML / CSS
Html5实现首页动态视频背景的示例代码
2019/09/25 HTML / CSS
一家专门经营包包的英国网站:MyBag
2019/09/08 全球购物
Hobbs官方网站:英国奢华女性时尚服装
2020/02/22 全球购物
全球最大化妆品零售网站:SkinStore
2020/10/24 全球购物
智乐游戏测试笔试题
2014/05/21 面试题
单身证明范本
2015/06/15 职场文书
新员工入职感想
2015/08/07 职场文书
2016年国陪研修感言
2015/11/18 职场文书
企业文化学习心得体会
2016/01/21 职场文书
2019年幼儿园管理条例范本!
2019/07/17 职场文书
教你部署vue项目到docker
2022/04/05 Vue.js
windows系统搭建WEB服务器详细教程
2022/08/05 Servers