攻克javascript闭包面试难点

闭包的定义

在计算机科学中对闭包的定义(摘自维基百科)
  • 闭包(英文:Closure),又称词法闭包(Lexical Closure)或函数闭包(function closure)。

在解析阶段有一个词法解析,是在编译之前进行的,也就意味着在解析阶段已经确定了闭包的执行上下文

  • 是在支持头等函数的编程语言中,实现词法绑定的一种技术。

像javascript这样的编程语言,在函数中能够声明函数,称为头等函数

  • 闭包在实现上是一个结构体,它存储了一个函数和一个关联的环境(相当于是一个符号查找表)
  • 闭包跟函数最大的区别是,当捕捉闭包的时候,它的自由变量会在捕捉的时候被确定,这样即使脱离了捕捉时的上下文,他也能照常运行。
MDN中对javascript闭包的解释

javascript中闭包的起源:
最早实现闭包程序是Scheme,而javascript中有大量设计来源于Scheme

  • 一个函数和对其周围的状态(lexical environment,词法环境)的引用捆绑在一起,这样的组合就是闭包(closure)
  • 也就是,闭包让你可以在一个内层函数中访问到外层函数作用域
  • 在javascript中,每当创建一个函数,闭包就在函数创建的时候同时被创建出来(广义角度)

简单闭包实例解释闭包形成过程

  function foo(){
    var yy='yy'
    function bar(){
      console.log(yy)
    }
    return bar
  }
  var fn=foo()
  fn()
  /*函数内层作用域中访问到了外层作用域,显然形成了闭包*/
V8引擎执行javascript代码过程

(不了解可以看一下之前我写的文章哦)

  • 代码解析过程
    在这里插入图片描述
  • 执行到 var fn = foo() 时候,(也就是foo函数执行时候)
  • 在这里插入图片描述
  • var fn = foo() 执行结束 函数执行上下文消失,fn=x0b00

在这里插入图片描述

  • 代码执行到fn()时候(相当于bar函数执行),同样创建属于bar函数的AO,和执行上下文函数
    在这里插入图片描述

执行bar函数的时候,console.log(yy) 语句执行,根据作用域链,会先在bar的AO里边找yy变量,发现没有,找到了父级作用域(parent scope)的指向 foo 的AO,然后输出结果“yy”

  • 代码全部执行完毕

这时候bar的函数执行上下文消失,由于AO(bar)没有根管理,所以最后所占内存也会被自动销毁掉

在这里插入图片描述

总结

通过上边演示闭包内存变化状态,我们可以看出,代码执行结束,内存是这样的样子
在这里插入图片描述

由于AO(foo)有根管理着(fn:指向的 x0b00是关键),所以内存也处于未被清理状态,如果我们不再使用这些内存(bar函数只被调用一次),而这些所占的空间也没有被清理掉,那么我们就称之为内存泄漏

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值