透过Extjs学习JavaScript---闭包篇

目录

一、前言
四、总结

一、前言

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秒的缓冲期内重复调用就不会重复执行函数。

四、总结

  1. 笔者的案例代码是取自于ExtJs,可能有些不好理解,但是当你看懂了,就能感觉到JS这门语言的巧妙。
  2. 文本只是通过介绍闭包的方法,进而介绍下ExtJs的源码,使我们撸代码的水平有所提升。

五、常见问题

  • 问:有的人会问这创建延迟函数有什么用?感觉在实际情况下用不到? 
    答:笔者可不这么认为,我在做项目中就常常用到这种方法。当我们要对一个输入框发生内容改变时,进行AJAX查询。但又不想每输入一个字就进行一次查询,因为这样非常的消耗服务器资源。较好的解决方案是设置一个输入缓冲期,在这个输入缓冲期内输入字母不进行查询,等过了缓冲期在对之前输入过的所有字母进行查询,所以我们需要一个“带有延迟缓冲功能”的函数。
  • 问:arguments这到底是什么? 
    答:好吧,详细请访问http://www.w3school.com.cn/js/pro_js_functions_arguments_object.asp,arguments是一个特殊的对象,类似于数组,但不完全是数组,所以要用“Array.prototype.slice.call(arguments, 0)”方法转成常规的数组。
posted @ 2014-02-19 11:15  小明你怎么了  阅读(1218)  评论(0编辑  收藏  举报