JavaScript学习笔记 - 闭包

我觉得在理解闭包之前,先要理解有两点:

  1. 作用域
  2. JavaScript的垃圾回收机制

作用域就不用多说了,必须懂得基础知识,简单的说下垃圾回收机制:

根据MDN的文档,目前主要的垃圾回收算法有两种:

  1. 引用-计数,简单的说就是找到所有没有被引用到的对象,把它们回收,但是这个算法存在一个问题,就是无法处理循环引用(a引用b,b引用a)
  2. 标记-清除,标记就是从根对象开始(在浏览器环境中,JavaScript的根对象就是window),找所有从根开始引用的对象,再找这些对象引用的对象...,所有能够查询到的对象都不会被清除,查询不到的对象全部被回收。(这也是目前浏览器使用的算法)

垃圾回收的目的是为了释放那些无用的变量占用的内存

参考资料: 内存管理 JavaScript变量、作用域、垃圾回收机制和闭包理解

了解了垃圾回收机制后,我们就知道在全局变量是不会被回收的(只有页面关闭全局变量才会被回收)。因为在全局作用域中声明一个函数或者变量,相当于给全局对象window添加一个属性,例子:

现在,来说说闭包。

闭包的特点:

  • 闭包是一个函数
  • 闭包可以使函数外部的作用域访问函数内部的作用域
  • 闭包可以阻止函数在执行完后的内部变量不被回收

例子:

function foo() {
  let a = 0
  function fn () {
     a += 1
     console.log(a)
  }
  return fn
}
let bar = foo()
bar()  // 1
bar()  // 2
bar()  // 3
复制代码

函数foo返回了一个函数,这个函数由于作用域的原因是可以访问到函数foo的作用域的,把这个函数赋值给全局变量bar,这样bar就可以在函数foo的作用域外去修改foo内部变量a的值,由于bar是全局变量,所以它不会被回收,而它又引用了fn函数,fn函数又引用了foo中的变量a,导致foo在执行完成之后,它内部的变量没有被回收,这就是一个最简单的闭包了。

  • 闭包的作用

    • 模块模式 模块模式是一种设计模式,就是将为了实现某个功能的函数或者变量都放在一起。这样做的好处有:

      1. 方便维护
      2. 避免命名冲突
      3. 复用性更高

例子:

function control() {
  let currentSpeed = 0;

  function setSpeed(speed) {
    return currentSpeed = speed;
  }

  function getSpeed() {
    return currentSpeed;
  }

  function accelerate(speed) {
    return currentSpeed += speed;
  }

  function decelerate(speed) {
    return currentSpeed -= speed;
  }

  function getStatus() {
    if (currentSpeed > 0) {
      return 'running';
    } else {
      return 'stop'
    }
  }
  return {
    setSpeed,
    getSpeed,
    accelerate,
    decelerate,
    getStatus
  }
}

const car = control()
复制代码

这样就封装了一个简单的模块,该模块暴露的方法有setSpeed,getSpeed等等,就可以通过这些方法去改变control函数中currentSpeed变量。

总结下就是,在函数中通过返回一个新函数,能使函数外部的作用域访问到函数内部的作用域,就产生了闭包。

转载于:https://juejin.im/post/5b90d2255188255c8138ce92

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值