JS 之 作用域及闭包

89 阅读4分钟

JS 之 作用域及闭包

在 JavaScript 中,作用域是指变量在代码中可访问的范围,离开了这个“范围”变量便不能被访问。

本文学习目标:了解作用域对程序执行的影响及作用域链的查找机制,使用闭包函数创建隔离作用域避免全局变量污染。

作用域分为:全局作用域局部作用域

1、全局作用域

script 标签 和 .js 文件 的【最外层】就是所谓的全局作用域,在此声明的变量在函数内部也可以被访问。全局作用域中声明的变量,任何其它作用域都可以被访问

<script>
  /* 全局作用域 */
  const global = 10; // 声明的全局变量global

  function func() {
    console.log(global)// 函数内部可以使用全局变量
  }
</script>

注意:
  1、为 window 对象动态添加的属性默认也是全局的,不推荐!!!
  2、函数中未使用任何关键字声明的变量为全局变量,不推荐!!!
  3、尽可能少的声明全局变量,防止全局变量被污染

2、局部作用域

局部作用域分为:函数作用域块作用域

2.1 函数作用域

在函数内部声明的变量只能在函数内部被访问,外部无法直接访问。

<script>

  function func() {
    const localityNum = 10; 
  }
  console.log(localityNum)// Uncaught ReferenceError: localityNum is not defined 函数外部不能使用局部作用域变量
</script>

注意:
  1、函数内部声明的变量,在函数外部无法被访问。
  2、函数的参数也是函数内部的局部。
  3、不同函数内部声明的变量无法互相访问。
  4、函数执行完毕后,函数内部的变量实际被清空。
2.1 块作用域

在 JavaScript 中使用 { } 包裹的代码称为代码块,代码块内部声明的变量外部将【有可能】无法被访问。

<script>

  for (let i=0; i<10; i++) {
    // i只能在该代码块中执行
    console.log( i );// 正常
  }
  console.log( i ); // ReferenceError 超出i的作用域
</script>

注意:
  1let 声明的变量会产生块作用域,var 不会产生块作用域。
  2const 声明的常量也会产生块作用域。
  3、不同代码块之间的变量无法互相访问。
  4、推荐使用 letconst创建变量。

3、作用域链

作用域链本质上是底层的变量查找机制。

在函数被执行时,会优先查找当前函数作用域中查找变量。

如果当前作用域查找不到则会依次逐级查找父级作用域直到全局作用域。


Global Execution Context
   |
   +-- Function Execution Context 1
   |      |
   |      +-- Function Execution Context 2
   |             |
   |             +-- Function Execution Context 3
   |
   +-- Function Execution Context 4
   
 

![作用域链.png](https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/76eacd3ad05546498dbd73851514cf70~tplv-k3u1fbpfcp-jj-mark:0:0:0:0:q75.image#?w=1151&h=440&s=155182&e=jpg&b=fdfdfd)


总结:
1、 嵌套关系的作用域串联起来形成了作用域链
2、 相同作用域链中按着从小到大的规则查找变量
3、 子作用域能够访问父作用域,父级作用域无法访问子级作用域

4、闭包

MDN定义:闭包(closure)是一个函数以及其捆绑的周边环境状态(lexical environment,词法环境)的引用的组合。换而言之,闭包让开发者可以从内部函数访问外部函数的作用域。在 JavaScript 中,闭包会随着函数的创建而被同时创建。

简单理解:闭包 =  内层函数 + 外层函数的变量

function createCounter() {  
  let count = 0; // 局部变量  
    
  return function() { // 返回的函数是闭包  
    count += 1; // 可以访问并修改外部函数的变量  
    return count;  
  };  
}  
  
let counter = createCounter(); // 创建闭包  
console.log(counter()); // 输出 1  
console.log(counter()); // 输出 2  
console.log(counter()); // 输出 3

在这个例子中,createCounter 函数返回一个匿名函数,这个匿名函数可以访问并修改 createCounter 函数中的 count 变量。即使 createCounter 函数已经执行完毕,这个匿名函数仍然可以访问并修改 count 变量。这就是闭包的基本概念。

闭包的优点:

1、可以避免全局变量的污染,有助于维护程序的安全性和稳定性。

2、可以延长变量的生命周期,即使外部函数已经执行完毕,闭包仍然可以访问并修改外部函数的变量。

3、可以将函数与相关引用变量集合在一起,方便调用和访问。

闭包的缺点:

1、可能会导致内存浪费问题。由于闭包会保持对其引用变量的引用,因此如果不及时释放引用,会导致内存泄漏。

2、在低版本IE浏览器中可能会导致内存泄露。解决方法是在退出函数之前,将不使用的局部变量全部删除。

闭包学习推荐:现代JavaScript高级教程小册