JavaScript高级程序设计 学习笔记 js高级技巧


Posted in Javascript onSeptember 20, 2011

第十八章 高级技巧
1.高级函数
1.1 作用域安全的构造函数
①直接调用构造函数而不适用new操作符时,由于this对象的晚绑定,它将映射在全局对象window上,导致对象属性错误增加到window。

function Person(name,age,job){ 
this.name = name; 
this.age = age; 
this.job = job; 
} 
Var person = Person("Jay",29,"singer"); //属性增加到window对象上。

②作用域安全构造函数
function Person(name,age,job){ 
if(this instanceof Person){ 
this.name = name; 
this.age = age; 
}else{ 
return new Person(name,age); 
} 
}

③上述作用域安全的构造函数,如果使用构造函数窃取模式的继承且不使用原型链,那么这个继承很可能被破坏。
□如果构造函数窃取结合使用原型链或者寄生式组合则可以解决这个问题。
function Polygon(side){ 
if(this instanceof Polygon){ 
this.sides = sides; 
this.getArea = function{return 0;}; 
}else{ 
return new Polygon(sides); 
} 
} 
function Rectangle(width,height){ 
Polygon.call(this,2); 
this.width = width; 
this.height = height; 
this.getArea = function(){ 
return this.width * this.height; 
}; 
} 
Rectangle.prototype = new Polygon(); 
var rect = new Rectangle(5,10); 
alert(rect.sides); //2

1.2 惰性载入函数
①惰性载入表示函数执行的分支仅会发生一次:既第一次调用的时候。在第一次调用的过程中,该函数会被覆盖为另一个按合适方式执行的函数,这样任何对原函数的调用都不用再经过执行的分支了。
■优点:
□要执行的适当代码只有当实际调用函数时才进行。
□尽管第一次调用该函数会因额外的第二个函数调用而稍微慢点,但后续的调用都会很快,因避免了多重条件。
function create XHR(){ 
if(typeof XMLHttp Request != "undefined"){ 
createXHR = function(){ 
return new XMLHttpRequest(); 
}; 
}else if(typeof ActiveXObject != "undefined"){ 
createXHR = function(){ 
if(typeof arguments.callee.activeXString != "string"){ 
var versions = ["MSXML2.XMLHttp.6.0","MSXML2.XMLHttp.3.0","MSXML2.XMLHttp"]; 
for(vai I = 0, len = versions.length; I < len; i++){ 
try{ 
Var xhr = new ActiveXObject(version[i]); 
Arguments.callee.activeXString = version[i]; 
Return xhr; 
}catch(ex){ 
//skip 
} 
} 
} 
return new ActiveXObject(arguments.callee.activeXString); 
}; 
}else{ 
createXHR = function(){ 
throw new Error("No XHR Object available."); 
}; 
} 
return createXHR(); 
}

1.3 函数绑定
①函数绑定要创建一个函数,可以在特定环境中以指定参数调用另一个函数。
②一个简单的bind()函数接受一个函数和一个环境,并返回一个在给定环境中调用给定函数的函数,并且将所有参数原封不动传递过去。
function bind(fn, context){ 
return function(){ 
return fn.apply(context, arguments); 
}; 
}

③被绑定函数与普通函数相比有更多的开销——它们需要更多内存,同时也因为多重函数调用而稍微慢一点——所以最好只在必要时使用。
1.4 函数柯里化
定义:用于创建已经设置好了一个或多个参数的函数。函数柯里化的基本方法和函数绑定是一样的:使用一个闭包返回一个函数。两者的区别在于,当函数被调用时,返回函数还需要设置一些传入的参数。
function bind(fn, context){ 
var args = Array.prototype.slice.call(arguments, 2); 
return function(){ 
var innerArgs = Array.prototype.slice.call(arguments); 
var finalArgs = args.concat(innerArgs); 
return fn.apply(context,finalArgs); 
}; 
}

2.高级定时器
①JavaScript是单线程程序,定时器是在间隔时间后将代码添加到列队。
②执行完一套代码后,JavaScript进程返回一段很短的时间,这样页面上的其他处理就可以进行了。
2.1 重复的定时器
①setInterval()仅当没有该定时器的任何其他代码实例时,才将定时器代码添加到队列中。
□某些间隔会被跳过。
□多个定时器代码执行之间的间隔可能会比预期小。
②避免setInterval()的两个缺点,使用链式setTimeout()调用:
setTimeout(function(){ 
//处理 
if(condition){ 
setTimeout(arguments.callee, interval); 
} 
},interval);

2.2 Yielding Processes
①JavaScript长时间运行脚本制约:如代码运行超过特定的时间或特定的语句数量就不会让它继续执行。
②当某个函数要花200ms以上的事件完成,最好分割为一系列可以使用定时器的小任务。
③数组分块技术:为要处理的项目创建一个队列,然后使用定时器取出下一个要处理的项目进行处理,接着再设置另一个定时器。
function chunk(array, process, context){ 
setTimeout(function(){ 
var item = array.shift(); 
process.call(context,item); 
if(array.length>0){ 
setTimeout(arguments.callee, 100); 
} 
} 
}

2.3 函数节流
①DOM操作比起非DOM交互需要更多内存和CPU时间。连续尝试进行过多的DOM相关操作可能会导致浏览器挂起,有时甚至崩溃。
②函数节流思想:某些代码不可以在没有间断的情况连续重复执行。
□示例
var processor = { 
timeoutId : null, 
//实际进行处理的方法 
performProcessing : function(){ 
//实际执行的方法 
}, 
//初始处理调用的方法 
process : function(){ 
clearTimeout(this.timeoutId); 
var that = this; 
this.timeoutId = setTimeout(function(){ 
that.performProcessing(); 
},100); 
} 
}; 
//尝试开始执行 
Processor.process(); 
□简化模式 
function throttle(method,context){ 
clearTimeout(mehtod.tId); 
mehtod.tId = setTimeout(function(){ 
method.call(context); 
},100); 
}

3.自定义事件
①事件是一种叫做观察者的设计模式,这是一种创建松散耦合代码的技术。
□对象可以发布事件,用来表示该对象声明周期中某个有趣的时刻到了。
□其他对象可以观察该对象,等待有趣的时刻到来并通过运行代码来响应。
②观察者模式由两类对象组成:主体和观察者。
□主体负责发布事件,同时观察者通过订阅这些事件来观察主体。
□主体并不知道观察者的任何事情,它可以独立自存在并正常运作即使观察者不在。
③自定义事件:创建一个管理事件的对象,让其他对象监听那些事件。
function EventTarget(){ 
this.handlers = {}; 
} 
EventTarget.prototype = { 
constructor : EventTarget, 
addHandler : function(type,handler){ 
if(typeof this.handlers[type] == "undefined"){ 
this.handlers[type] = []; 
} 
this.handlers[type].push(handler); 
}, 
fire : function(event){ 
if(!event.target){ 
event.target = this; 
} 
if(this.handlers[event.type] instanceof Array){ 
var handlers = this.handlers[event.type]; 
for(var i=0,len=handlers.length; i<len; i++){ 
handlers[i](event); 
} 
} 
}, 
removeHandler : function(type, handler){ 
if(this.handlers[type] instanceof Array){ 
var handlers = this.handlers[type]; 
for(var i=0,len=handlers.length; i<len; i++){ 
if(handlers[i] === handler){ 
break; 
} 
} 
Handlers.splice(i,1); 
} 
};

④使用EventTarget类型的自定义事件可以如下使用:
function handleMessage(event){ 
alert("message received:" + event.message); 
} 
//创建一个新对象 
var target = new EventTarget(); 
//添加一个事件处理程序 
target.addHandler("message",handleMessage); 
//触发事件 
target.fire({type:"message",message:"hello world!"}); 
//删除事件处理程序 
target.removeHandler("message",handleMessage);

⑤使用实例
function Person(name,age){ 
eventTarget.call(this); 
this.name = name; 
this.age = age; 
} 
inheritPrototype(Person, EventTarget); 
Person.prototype.say = function(message){ 
this.fire({type:"message", message:message}); 
}; 
function handleMessage(event){ 
alert(event.target.name + "says: " + event.message); 
} 
//创建新person 
var person = new Person("Nicholas",29); 
//添加一个事件处理程序 
Person.addHandler("message",handleMessage); 
//在该对象上调用1个方法,它触发消息事件 
person.say("Hi there");

4.拖放
功能:①拖放②添加了自定义事件
var DragDrop = function(){ 
var dragdrop = new EventTarget(); 
var dragging = null; 
var diffX = 0; 
var diffY = 0; 
function handleEvent(event){ 
//获取事件和对象 
event = EventUtil.getEvent(event); 
var target = EventUtil.getTarget(event); 
//确定事件类型 
switch(event.type){ 
case "mousedown" : 
if(target.className.indexOf("draggable")>-1){ 
dragging = target; 
diffX = event.clientX - target.offsetLeft; 
diffY = event.clientY - target.offsetTop; 
dragdorp.fire( 
{ 
type:"dragstart", 
target : dragging, 
x : event.clientX, 
y : event.clientY 
} 
); 
break; 
case "mousemove" : 
if(dragging !== null){ 
//获取事件 
event = EventUtil.getEvent(event); 
//指定位置 
dragging.style.left = (event.clientX - diffX) + "px"; 
dragging.style.top = (event.clientY - diffY) + "px"; 
//触发自定义事件 
dragdrop.fire( 
{ 
type : "drag", 
target : dargging, 
x : event.clientX, 
y : event.clientY 
} 
); 
} 
break; 
case "mouseup" : 
dargdorp.fire( 
{ 
type : "dragend", 
target : dragging, 
x : event.clientX, 
y : event.clientY 
} 
); 
dragging = null; 
break; 
} 
} 
//公共接口 
dragdrop.enable = function() { 
EventUtil.addHandler(document, "mousedown", handleEvent); 
EventUtil.addHandler(document, "mousemove", handleEvent); 
EventUtil.addHandler(document, "mouseup", handleEvent); 
}; 
dragdrop.disable = function(){ 
EventUtil.removeHandler(document, "mousedown", handleEvent); 
EventUtil.removeHandler(document, "mousemove", handleEvent); 
EventUtil.removeHandler(document, "mouseup", handleEvent); 
}; 
return dragdrop; 
}();
Javascript 相关文章推荐
一个js实现的所谓的滑动门
May 23 Javascript
js中top/parent/frame概述及案例应用
Feb 06 Javascript
javascript函数作用域学习示例(js作用域)
Jan 13 Javascript
js实现的map方法示例代码
Jan 13 Javascript
html中鼠标滚轮事件onmousewheel的处理方法
Nov 11 Javascript
jQuery网页定位导航特效实现方法
Dec 19 Javascript
js实现自动轮换选项卡
Jan 13 Javascript
JS去掉字符串前后空格或去掉所有空格的用法
Mar 25 Javascript
实现单层json按照key字母顺序排序的示例
Dec 06 Javascript
vue.js实现插入数值与表达式的方法分析
Jul 06 Javascript
关于自定义Egg.js的请求级别日志详解
Dec 12 Javascript
JSON字符串操作移除空串更改key/value的介绍
Jan 05 Javascript
跨浏览器通用、可重用的选项卡tab切换js代码
Sep 20 #Javascript
JQuyer $.post 与 $.ajax 访问WCF ajax service 时的问题需要注意的地方
Sep 20 #Javascript
简单的jquery拖拽排序效果实现代码
Sep 20 #Javascript
用jQuery中的ajax分页实现代码
Sep 20 #Javascript
jquery模拟按下回车实现代码
Sep 20 #Javascript
一个分享按钮的插件使用介绍(可扩展,内附开发制作流程)
Sep 19 #Javascript
50个比较实用jQuery代码段
Sep 18 #Javascript
You might like
PHP实现补齐关闭的HTML标签
2016/03/22 PHP
php中yar框架实例用法讲解
2020/12/27 PHP
DWR Ext 加载数据
2009/03/22 Javascript
jquery实现居中弹出层代码
2010/08/25 Javascript
javascripit实现密码强度检测代码分享
2013/12/12 Javascript
使用Node.js配合Nginx实现高负载网络
2015/06/28 Javascript
基于jquery animate操作css样式属性小结
2015/11/27 Javascript
jQuery使用$.ajax进行异步刷新的方法(附demo下载)
2015/12/04 Javascript
vue2.0 根据状态值进行样式的改变展示方法
2018/03/13 Javascript
用Vue.js在浏览器中实现裁剪图像功能
2019/06/18 Javascript
JavaScript 作用域scope简单汇总
2019/10/23 Javascript
JS实现点击下拉列表文本框中出现对应的网址,点击跳转按钮实现跳转
2019/11/25 Javascript
原生js实现日历效果
2020/03/02 Javascript
js实现批量删除功能
2020/08/27 Javascript
Vue实现简易购物车页面
2020/12/30 Vue.js
[13:18]《一刀刀一天》之DOTA全时刻21:详解TI新赛制 A队再露獠牙
2014/06/24 DOTA
centos下更新Python版本的步骤
2013/02/12 Python
Python和JavaScript间代码转换的4个工具
2016/02/22 Python
浅谈pyhton学习中出现的各种问题(新手必看)
2017/05/17 Python
tensorflow 只恢复部分模型参数的实例
2020/01/06 Python
Python3的socket使用方法详解
2020/02/18 Python
Matplotlib自定义坐标轴刻度的实现示例
2020/06/18 Python
Python项目打包成二进制的方法
2020/12/30 Python
T3官网:头发造型工具
2019/12/26 全球购物
企业面试题试卷附带答案
2015/12/20 面试题
SOA的常见陷阱或者误解是什么
2014/10/05 面试题
javascript实现用户必须勾选协议实例讲解
2021/03/24 Javascript
个人近期表现材料
2014/02/11 职场文书
《风筝》教学反思
2014/04/10 职场文书
班干部演讲稿
2014/04/24 职场文书
企业领导班子四风对照检查材料
2014/09/27 职场文书
师范生见习报告范文
2014/11/03 职场文书
孔繁森观后感
2015/06/10 职场文书
浅谈Python数学建模之数据导入
2021/06/23 Python
SQL实现LeetCode(197.上升温度)
2021/08/07 MySQL
python 使用pandas读取csv文件的方法
2022/12/24 Python