目录
一、前言
JavaScript设计得最出色的就是它的函数的实现,它几乎接近于完美。我们现在现就来介绍它其中一个功能“闭包”。我们可以利用闭包“保存变量生命周期”和“屏蔽变量”的特性优雅地完成一些强大的功能。后面我还会介绍在ExtJS中是如何使用这种特性的。
二、基础讲解
我们可以利用“闭包”功能来实现隐藏或公开对象的变量和方法。
测试代码如下
1 var bufferObj = (function createFun(fn, interval, scope) { 2 var isExec = 1;//被闭包的变量,外界不能直接访问 3 return {//公开set,和exec方法 4 set : function(data) { 5 isExec = data; 6 }, 7 exec : function(fn) { 8 if (isExec) {//读取闭包的变量 9 console.log("函数可以执行"); 10 fn(); 11 } else { 12 console.log("函数不能执行"); 13 } 14 } 15 }; 16 })();
注意我们把整个函数用括号包住,并在最后一行加了"()",这样写法的意思是马上执行函数,并返回了结果值。这样我们的变量“bufferObj”对象就是带有公开的set和exec方法和私有变量“isExec”。我不能直接访问到“isExec”这个变量,这就是我们使用闭包的结果,将一些变量或方法优雅的变成私有,很好的控制了变量的作用域。
三、知识应用
上节初步的说明了闭包的写法和功能,看似功能不过如此,其实不然。下面我们来看看ExtJs是如何使用闭包功能的,希望可以给大家在使用闭包有更多的启发。这也是学习JS这门语言有兴趣的地方,是把一个简简单单的知识点写出多种花样。
下面的代码是extjs中“Ext.Function”类中的“createBuffered”方法,主要功能就是生成“带有延迟功能的函数”。我为了简化“Ext.Function”类的上下文,并把代码标上的汉语解释方便讲解。
Ext.Function 代码如下:
1 var Ext = {}; 2 Ext.Function = { 3 /** 4 * 返回带有延迟功能的函数,如果在延迟的时间内再次调用该函数,重置缓冲时间 5 * @param {Function} fn 需要被延迟的方法 6 * @param {Number} buffer 延迟时间(毫秒) 7 * @param {Object} [scope=this] 传入方法的作用域 8 * @param {Array} [args] 方法参数 9 * @return {Function} 返回带有缓冲功能的函数 10 */ 11 createBuffered: function(fn, buffer, scope, args) { 12 var timerId;//计时器ID 13 14 return function() { 15 var callArgs = args || Array.prototype.slice.call(arguments, 0),//将“arguments”类型转成数组类型 16 me = scope || this;//如果没有传入“scope”参数就全用当前上下文的"this"为作用域 17 18 if (timerId) {//如果有计时器,就取消之前的计时器 19 clearTimeout(timerId); 20 } 21 22 timerId = setTimeout(function(){//重置缓冲时间 23 fn.apply(me, callArgs);//在相应的作用域上执行方法 24 }, buffer); 25 }; 26 } 27 };
在上述代码中把变量“timerId”闭包在对象内,只返回了一个带有缓冲功能的方法,该返回方法主要是判断函数在缓冲期内是否被重复调用,如果在缓冲期内被调用就重置计时器。
在代码中有三个知识点:
1. 15行中“Array.prototype.slice.call(arguments, 0)”这句意思是把arguments参数类型转成数组类型,方便第23行的“apply”调用。(如果不知道arguments是什么意思的同学就自行面壁思过吧,哈哈哈。)
2. 16行中"me = scope || this" ,是一个if判断取巧的写法。整句的意思是,当“scope”为“空”或“假”时就把“this”返回给“me”变量。
3. 18行中的“timerId”变量由于是被闭包,他的“生命期”被延长了,不会随着返回函数的运行结束而结束,所以每次调用返回函数时,都可以保留之前的“timerId”值。
使用代码如下:
1 var run = function(data) { 2 alert("函数运行成功"); 3 alert("入参为:" + data); 4 }; 5 var bufferFn = Ext.Function.createBuffered(run, 3000); 6 bufferFn("JavaScript1");//不会被执行,因为后一个函数在缓冲期3秒内调用,把这当前的函数取消了 7 bufferFn("JavaScript2");//不会被执行,因为后一个函数在缓冲期3秒内调用,把这当前的函数取消了 8 bufferFn("JavaScript3");//会被执行,因为后一个函数没有在缓冲期3秒内调用,所以该函数会执行 9 10 setTimeout(function() {//间隔4秒 11 bufferFn("JavaScript4");//会被执行,因为后面没有重复调用了 12 }, 4000);
在上述代码第5行,就是通过“Ext.Function.createBuffered”生成了一个带有缓冲功能的函数“bufferFn”,如果我们在3秒的缓冲期内重复调用就不会重复执行函数。
四、总结
- 笔者的案例代码是取自于ExtJs,可能有些不好理解,但是当你看懂了,就能感觉到JS这门语言的巧妙。
- 文本只是通过介绍闭包的方法,进而介绍下ExtJs的源码,使我们撸代码的水平有所提升。
五、常见问题
- 问:有的人会问这创建延迟函数有什么用?感觉在实际情况下用不到?
答:笔者可不这么认为,我在做项目中就常常用到这种方法。当我们要对一个输入框发生内容改变时,进行AJAX查询。但又不想每输入一个字就进行一次查询,因为这样非常的消耗服务器资源。较好的解决方案是设置一个输入缓冲期,在这个输入缓冲期内输入字母不进行查询,等过了缓冲期在对之前输入过的所有字母进行查询,所以我们需要一个“带有延迟缓冲功能”的函数。 - 问:arguments这到底是什么?
答:好吧,详细请访问http://www.w3school.com.cn/js/pro_js_functions_arguments_object.asp,arguments是一个特殊的对象,类似于数组,但不完全是数组,所以要用“Array.prototype.slice.call(arguments, 0)”方法转成常规的数组。