js闭包

 已下是自己阅读文档js高级程序设计后的知识点,在此记录,以便于之后温习

闭包

1.闭包是指有权访问另一个函数作用域中变量的函数。常见方式:在另一个函数中创建另一个函数(匿名)。

闭包是一个函数,但是它创建在另一个函数内部。闭包关键在于它的执行环境和它的作用域链。

 

2.当函数被调用,会创建一个执行环境和作用域链,并将作用域链赋给函数的一个内部属性[[Scope]]。然后用this,arguments,和其他命名属性来初始化活动对象,并把这个活动对象添加在作用域链前端。外部函数的活动对象位于作用域链第二位,再外层的位于第三位,直到作用域链的最外层为全局的变量对象。

 

 

3.一般情况下,函数执行完会销毁活动对象,有闭包的话情况会有所不同,在外层函数执行完毕后,闭包还在使用外层函数的活动对象,那么外层活动对象不会被销毁,将继续保存在内存中,直到闭包代码执行完毕,一起销毁。

5.闭包与变量

作业域链本身是一个指针列表,只引用但不实际包含变量对象。用如下代码来理解

function createFunctions(){
            var result=new Array();

            for(var i=0;i<10;i++)
            {
                result[i]=function(){
                    return i;
                }
            }
            
            for(let i=0;i<10;i++)
            {
                console.log(result[i]());
            }
            //return result;
        }
        createFunctions();

这个代码表面上看会打印回索引值:0,1,2,3...但是实际上它打印的是10个10。

原因:作业域链本身是一个指针列表,只引用但不实际包含变量对象。result数组是函数数组,每个元素都是一个匿名函数,而每个函数的作用域链都保存着createFunctions的活动对象,即每个函数都在 引用 同一个变量i,所以当调用result数组中的元素函数时,返回的都是同样的值,i在循环时最后累加为10。

上面的例子也可以看出闭包的副作用:闭包只能取包含函数中任何变量的最后一个值。

 

6.this对象

js中的this已经很迷了,闭包与this碰在一起就更加迷上迷?!

其实还是比较简单:匿名函数的执行环境具有全局性,所以闭包中的this对象通常指向window

var name="this is window";
        var obj={
            name:"this is obj",
            getName:function(){
                return function(){
                    return this.name;
                };
            }
        }
        alert(obj.getName()());//this is window

 

函数调用时,活动对象会自动获取this和argurments,不会直接访问外部函数的这两个变量,不过可以将外部作用域的this保存在闭包能访问的变量里面。

 

posted @ 2019-08-15 11:25  ellenxx  阅读(108)  评论(0编辑  收藏  举报