关于js闭包

关于闭包

简单来说,闭包就是函数内部定义的函数,被返回了出去并在外部调用。内部的函数存在外部作用域的引用就会导致闭包。

闭包会导致作用域链不被释放,作用域中的变量参数会被保存下来。最大的作用域全局作用域只有页面关闭才会释放。
变量查询是从当前函数的作用域链中从顶部依次向下查询。代码执行的环境和基本类型值存储在栈内存中,函数执行完,但是函数的私有作用域内有内容被栈外的变量还在使用的,栈内存就不能释放里面的基本值也就不会被释放。所以作用域链不被释放。

        function a(){
            var num = 100;
            function b(){
                num ++;
                console.log(num);
            }
            return b;
        }
        var demo = a();
            demo()
            demo()
            demo()
            demo()

函数执行四次,结果分别为
在这里插入图片描述
可以看出每次执行改变的结果都被保存下来了。

作用域链不被释放,什么是作用域链?
作用域链:[[scopes]]中所存储的执行上下文的集合,以链接结构进行存储。

执行上下文:当函数运行时,会创建一个称为执行上下文的内部对象,一个执行上下文定义了函数运行时的环境,函数每次执行时都会创建一个独一无二的执行上下文,并且在函数执行完毕之后,会销毁掉函数所产生的执行上下文。

[[scopes]]中存储:[0:GO]或[0:AO,1:GO]或[0:AO,1:AO,2:GO]。第0位存放的永远是自己的上下文。
变量查询是从当前函数的作用域链中从顶部依次向下查询,就是从第0位向下查询。

预编译

  • 全局预编译
    a . 会生成一个(Gloabl Object) GO对象
    b . 分析var变量声明,如果变量在GO对象中不存在,直接将变量名作为AO对象的属性名,值为undefined,如果已经存在,不做任何改变
    c . 分析function函数,将函数名作为GO对象的属性名,值为函数体,如果已存在,直接覆盖
  • 局部预编译
    a . 函数在运行的一瞬间,会产生一个(Active Object) AO对象
    b . 函数声明的参数(形参),成为AO对象的属性,属性值为实参的值,如果没有传递实参 属性值为undefined
    c . 分析var变量声明,如果变量在AO对象中不存在,直接将变量名作为AO对象的属性名,值为undefined,如果已经存在,不做任何改变
    d . 分析function函数,将函数名作为AO对象的属性名,值为函数体,如果已存在,直接覆盖。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值