学习JavaScript设计模式之代理模式
- 明星都有经纪人作为代理。如果请明星办一场商演,只能联系其经纪人,经纪人会把商演的细节和报酬谈好,再把合同交给明星签。
一、定义
代理模式:为一个对象提供一个代用品或占位符,以便控制对它的访问。
代理分为:保护代理和虚拟代理
保护代理:用于控制不同权限的对象对目标对象的访问,在JavaScript中很难判断谁访问了某个对象,所以保护代理很难实现。
二、图片预加载(最常见的虚拟代理应用场景)
图片预加载是一种常用技术,如果直接给某个img标签节点设置src属性,由于图片过大或者网络不佳,图片的位置往往有段时间会有空白。常见的做法事先用一张loading图片占位,然后异步加载图片,待图片加载完成,把其填充到img节点里。
实现原理:
创建一个Image对象:vara=newImage();
定义Image对象的src:a.src=“xxx.gif”;
这样做就相当于给浏览器缓存了一张图片。
可通过Image对象的complete属性来检测图像是否加载完成。每个Image对象都有一个complete属性,当图像处于装载过程中时,该属性值false,当发生了onload、onerror、onabort中任何一个事件后,则表示图像装载过程结束,此时complete属性为true。
(1)非代理实现
varmyImage=(function(){
varimgNode=document.createElement("img");
document.body.appendChild(imgNode);
varimg=newImage();
img.onload=function(){
imgNode.src=img.src;
};
return{
setSrc:function(src){
imgNode.src="./images/loading.gif";
img.src=src;
}
}
})();
myImage.setSrc("./images/originImg.png");
(2)代理实现
//创建图片DOM
varmyImage=(function(){
varimgNode=document.createElement("img");
document.body.appendChild(imgNode);
return{
setSrc:function(src){
imgNode.src=src;
}
};
})();
//代理
varproxyImage=(function(){
varimg=newImage();
img.onload=function(){
myImage.setSrc(this.src);//this指向img!img加载完成后,将img.src传递给myImage
};
return{
setSrc:function(src){
myImage.setSrc("./images/loading.gif");//loading
img.src=src;
}
};
})();
proxyImage.setSrc("./images/originImg.png");
使用代理模式的好处:使每个函数功能单一,实现对象设计的“单一职责原则”!
三、文件同步
假设我们在做一个文件同步功能,当选中checkbox时候,它对应的文件就会被同步到另外一台服务器。
<body> <inputtype="checkbox"id="1"/>文件1 <inputtype="checkbox"id="2"/>文件2 <inputtype="checkbox"id="3"/>文件3 <inputtype="checkbox"id="4"/>文件4 <inputtype="checkbox"id="5"/>文件5 <inputtype="checkbox"id="6"/>文件6 </body>
没选中一个checkbox就同步一次,显然不太合理。因为在web开发中,最大的开销就是网络请求。
解决方案:通过一个代理函数来收集一段时间之内的请求,然后一次性发给服务器。
varsynchronousFile=function(id){
console.log("开始同步文件,id为:"+id);
};
varproxySynchonousFile=(function(){
varcache=[],//保存本次需要同步文件的id
timer;//定时器
returnfunction(id){
cache.push(id);
if(timer){
//不要覆盖已经启动的定时
return;
}
timer=setTimeout(function(){
synchronousFile(cache.join(","));
clearTimeout(timer);
timer=null;
cache.length=0;//清空缓存
},2000);
}
})();
varcheckboxs=document.getElementsByTagName("input");
for(vari=0,c;c=checkboxs[i];i++){
c.onclick=function(){
if(this.checked===true){
proxySynchonousFile(this.id);
}
}
}
四、缓存代理–计算乘积(序列一模一样)
varmult=function(){
varresult=1;
for(vari=0,l=arguments.length;i<l;i++){
result=result*arguments[i];
}
returnresult;
};
varproxyMult=(function(){
varcache={};//{"1,2,3":6}
returnfunction(){
varargs=Array.prototype.join.call(arguments,",");
if(argsincache){
returncache[args];
}
returncache[args]=mult.apply(this,arguments);
}
})();
console.log(proxyMult(1,2,3));
//改造:
varproxyFactory=function(fn){
varcache={};
returnfunction(){
varargs=Array.prototype.join.call(arguments,",");
if(argsincache){
returncache[args];
}
returncache[args]=fn.apply(this,arguments);
}
};
console.log(proxyFactory(mult)(1,2,3));
希望本文所述对大家学习javascript程序设计有所帮助。