读javascript高级程序设计16-几条函数小技巧
内容概要
- 作用域安全的构造函数
- 惰性载入函数
- 函数绑定
- 函数节流
一、作用域安全的构造函数
我们知道,当使用new操作符调用构造函数时,构造函数内部的this会指向新创建对象的实例。
function Person(name){
this.name=name;
}
var p=new Person('peter');
console.log(p.name);//结果:perter
但是,如果没有使用new操作符,而是将构造函数当作普通函数调用时,this会指向window对象。
var p1=Person('peter');
console.log(p1.name);//报错
console.log(window.name);//peter
因此,在构造函数中应该首先检查this是否为正确的类型实例,这种方式就是作用域安全的构造函数。
function Person(name) {
if (this instanceof
Person) { this.name = name; } else { return new
Person(name);
}
}
var p = new Person('peter');
console.log(p.name); //perter
var p1 = Person('peter');
console.log(p1.name); //perter
console.log(window.name); //报错
在该构造函数基础上,结合原型链的实现如下:
function Person(name) {
if (this instanceof Person) {
this.name = name;
}
else {
return new Person(name);
}
}
function Student(name,sno){
Person.call(this,name);
this.sno=sno;
}
Student.prototype
=new
Person();
var s=new Student('peter','N0015');
console.log(s.name);
通过作用域安全的构造函数,可以保证在缺少new操作符调用构造函数的时候在正确的执行环境中进行。
二、惰性载入函数
有些函数包含很多分支,每次调用时都会执行一遍if分支判断,但实际上可能每次走的都是相同分支。看下这个创建CORS对象的方法:
function createCORSRequest(method,url){
//创建XHR对象
var xhr = new XMLHttpRequest();
//启动请求
if("withCredentials" in xhr){
xhr.open(method,url,true);
}else if(typeof XDomainRequest!='undefined'){
xhr=new XDomainRequest();
xhr.open(method,url);
}else{
xhr=null;
}
return xhr;
}
var xhr1 = createCORSRequest('get', 'http://www.othersite.com/weather.ashx');
var xhr2 = createCORSRequest('get', 'http://www.othersite.com/articles.ashx');
实际上对于同一款浏览器每次创建对象时都会执行相同的分支,因此每次都进行if判断是多余的,使用惰性载入函数可以实现函数执行分支只发生一次。有两种常见方案:
方案一:在第一次调用函数时,根据分支结果,将该函数替换为另一个按合适的方式执行的函数。我们改进上面的函数如下:
function createCORSRequest(method, url) {
var xhr0 = new XMLHttpRequest();
if ('withCredentials' in xhr0) {
var xhr = new XMLHttpRequest();
createCORSRequest = function (method, url) {
xhr.open(method, url, true);
return xhr;
}
} else if (typeof XDomainRequest != 'undefined') {
createCORSRequest = function (method, url) {
var xhr = new XMLHttpRequest();
xhr = new XDomainRequest();
xhr.open(method, url);
return xhr;
}
} else {
createCORSRequest = function (method, url) {
return null;
}
}
return createCORSRequest(method, url);
}
这样改进后,每次执行时不必再执行多个if分支判断,而是直接执行替换后的简洁函数。它只会在第一次调用时有一些性能损失。
方案二:在声明函数时,就指定适当的函数。同样改进上面的例子:
var createCORSRequest=(function(method, url) {
var xhr0 = new XMLHttpRequest();
if ('withCredentials' in xhr0) {
return function (method, url) {
var xhr = new XMLHttpRequest();
xhr.open(method, url, true);
return xhr;
}
} else if (typeof XDomainRequest != 'undefined') {
return function (method, url) {
var xhr = new XMLHttpRequest();
xhr = new XDomainRequest();
xhr.open(method, url);
return xhr;
}
} else {
return function (method, url) {
return null;
}
}
})();
这种方案不同的是使用var声明函数,并为每个分支指定了匿名且自动执行的函数。这样,函数第一次加载的时候就确定了执行哪个分支的函数。
三、函数绑定
在执行回调函数或者事件处理程序时,常常需要把函数作为变量传递,此时我们需要保存函数的代码执行环境。函数绑定就是要创建一个函数,可以在特定的this环境中以指定参数调用另一个函数
var demo = {
message: 'hello world',
show: function () {
console.log(this.message);
}
}
var obj = document.getElementById('my-btn');
EventUtil.addHandler(obj, 'click', demo.show);//点击按钮结果undefined
其实我们可以利用闭包写法修复这个问题:
EventUtil.addHandler(obj, 'click', function (event) {
demo.show();
});
但是用太多的闭包看上去代码不够简洁。在很多js库中,都会定义bind()函数来实现函数绑定。
function bind(fn, context) {
return function () {
return fn.apply(context, arguments);
}
}
这个bind函数实现比较简单,两个参数分别为要调用的函数和函数执行环境,执行结果返回一个在指定执行环境调用函数的函数,它把其内部函数的arguments参数全部传递过去。调用方法如下:
EventUtil.addHandler(obj, 'click',bind(demo.show,demo));
四、函数节流
对于周期性执行的代码,应该进行函数节流。
例如下面的滚动事件:
window.onscroll = function(){
throttle(demo,window);
}
当页面滚动的时候,会持续的输出结果。这种高频率的响应,如果方法复杂一些的话会耗用还多的性能。对于这种情况我们可以进行如下优化:
//设置和消除定时器
function throttle(method,context){
clearTimeout(method.id);
method.id=setTimeout(function(){
method.call(context)
},100);
}
function demo(){
console.log(1);
}
window.onscroll = function(){
throttle(demo,window);
}
throttle方法有两个参数:要执行的方法和作用域。第一次调用时会创建定时器,后续调用时都是先消除已有的定时器,然后重新创建定时器。这样一来只有最后一次调用之后100ms后才会将相应的方法加入执行队列。
读javascript高级程序设计16-几条函数小技巧的更多相关文章
- 读javascript高级程序设计00-目录
javascript高级编程读书笔记系列,也是本砖头书.感觉js是一种很好上手的语言,不过本书细细读来发现了很多之前不了解的细节,受益良多.<br/>本笔记是为了方便日后查阅,仅作学习交流 ...
- 读javascript高级程序设计-目录
javascript高级编程读书笔记系列,也是本砖头书.感觉js是一种很好上手的语言,不过本书细细读来发现了很多之前不了解的细节,受益良多.<br/>本笔记是为了方便日后查阅,仅作学习交流 ...
- 读javascript高级程序设计08-引用类型之Global、Math、String
一.Global 所有在全局作用域定义的属性和方法,都属于Global对象. 1.URI编码: encodeURI():主要用于对整个URI编码.它不会对本身属于URI的特殊字符进行编码. encod ...
- 读javascript高级程序设计01-基本概念、数据类型、函数
一. javascript构成 1.javascript实现由三部分组成: ECMAScript:核心语言功能 DOM:文档对象模型,提供访问和操作网页内容的方法和接口 BOM:浏览器对象模型,提供与 ...
- 读Javascript高级程序设计第三版第六章面向对象设计--创建对象
虽然Object构造函数或者对象字面量都可以用来创建单个对象,但是缺点非常明显:使用同一接口创建很多对象,会产生大量重复代码. 工厂模式 1 function CreatePerson(name,a ...
- 读javascript高级程序设计02-变量作用域
一. 延长作用域链 有些语句可以在作用域前端临时增加一个变量对象,该变量对象在代码执行完成后会被移除. ①with语句延长作用域. function buildUrl(){ var qs=" ...
- 读javascript高级程序设计05-面向对象之创建对象
1.工厂模式 工厂模式是一种常用的创建对象的模式,可以使用以下函数封装创建对象的细节: function CreatePerson(name,age){ var p=new Object(); p.n ...
- 读javascript高级程序设计07-引用类型、Object、Array
一.引用类型 ECMAScript是支持面向对象的,可以通过引用类型描述一类对象所具有的属性和方法. 创建对象实例的方法时是用new 操作符加构造函数:var p=new Person(). 二.Ob ...
- 读javascript高级程序设计10-DOM
一.节点关系 元素的childNodes属性来表示其所有子节点,它是一个NodeList对象,会随着DOM结构的变化动态变化. hasChildNodes():是否有子节点. var headline ...
随机推荐
- MVC+EF 自定义唯一性验证
最近悟出来一个道理,在这儿分享给大家:学历代表你的过去,能力代表你的现在,学习代表你的将来. 十年河东十年河西,莫欺少年穷 学无止境,精益求精 本篇提供二种方法,希望大家喜欢 1.自定义验证属性 ...
- JAVA经典算法40题(1-20)
[程序1] 题目:古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第四个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少? 1.程序分析: 兔子的规律 ...
- find常用参数详解
find常用参数详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 在linux系统中,在init 3模式情况下都是命令行模式,这个时候我们想要找到一个文件的就得依赖一个非常好用的 ...
- jSP-13-其他
1. JAVAEE Ø Java平台版本 Java平台有3个版本: 适用于小型设备和智能卡的JavaME (Java Platform Micro Edition,Java 微型版) 适用于桌 ...
- 执行gem install linne时报错
由于linner安装实际上是从 rubygems.org 获得的,而其被墙,所以,需要寻找国内的镜像进行安装: 第一种方法: gem sources --remove https://rubygems ...
- ASP.NET下载远程图片保存到本地的方法、保存抓取远程图片
以下介绍两种方法:1.利用WebRequest,WebResponse 类 WebRequest wreq=WebRequest.Create("http://www.xueit.com/e ...
- 解决Failed to allocate memory: 8转
解决Failed to allocate memory: 8 昨天换了x64的Win7,发现在Eclipse上启动模拟器的时候存在问题,当设置的模拟器分辨率大于400×800的时候会出现 Failed ...
- :selected
描述: 查找所有选中的选项元素 HTML 代码: <select> <option value="1">Flowers</option> < ...
- Application.DoEvents():概念
When you run a Windows Form, it creates the new form, which then waits for events to handle. Each ti ...
- Android 网络框架 volley源码剖析
转载请注明出处: http://blog.csdn.net/guolin_blog/article/details/17656437 经过前三篇文章的学习,Volley的用法我们已经掌握的差不多了, ...