作用域链与闭包

作用域链

核心原理

  1. VO中包含一个额外的属性(假设为 source),该属性指向创建该VO的函数本身
  2. 每个函数在创建时,会有一个隐藏属性[[scope]],它指向创建该函数时的AO
  3. 当访问一个变量时,会先查找自身VO中是否存在,如果不存在,则依次查找[[scope]]属性。

某些浏览器会优化作用域链,函数的[[scope]]中仅保留需要用到的数据。

示例分析

js 复制代码
<!doctype html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>Document</title>
  </head>

  <body>
    <script>
      var g = 0;

      function A() {
        var a = 1;

        function B() {
          var b = 2;
          var C = function () {
            var c = 3;
            console.log(c, b, a, g);
          };
          C();
        }

        B();
      }

      A();
    </script>
  </body>
</html>

说明:

蓝线:VO中包含一个额外的属性(假设为 source),该属性指向创建该VO的函数本身

红线:每个函数在创建时,会有一个隐藏属性[[scope]],它指向创建该函数时的AO

作用域链:当访问一个变量时,会先查找自身VO中是否存在,如果不存在,则依次查找[[scope]]属性(蓝红线交替 )

html 复制代码
<!doctype html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>Document</title>
  </head>

  <body>
    <script>
      var count = 100;
      
      function A() {
        var count = 0;
        return function () {
          count++;
          console.log(count);
        };
      }

      var test = A();

      test();
      test();
      test();

      console.log(count);
    </script>
  </body>
</html>

执行至 22 行时,执行上下文栈中的情况:

执行至 24 行时,执行上下文栈中的情况:

执行至 25 行时,执行上下文栈中的情况:

执行 count++时,线找寻 test 执行上下文中的 VO 是否有 count,如果没有,则根据 source寻找到创建该 VO 的函数本身,然后通过[[scope]]属性寻找到下一个 VO,若有 count 则该 count++,否则再往下寻找。

闭包

重新理解闭包:

原理上理解:执行上下文已经销毁,但执行上下文中 VO 被保留下来,因为匿名函数中有一个[[scope]]属性还指向他。并没有被垃圾回收器回收。

表面上理解:内部函数使用外部函数的变量,内部函数被外部调用。

相关推荐
anyup1 小时前
全面重构的 uni-app 多平台上传组件,功能强到离谱!
前端·vue.js·uni-app
暗不需求1 小时前
告别“class 命名地狱”:从面向对象 CSS 到原子 CSS(Tailwind) 的思维跃迁
前端·css·react.js
幼儿园技术家1 小时前
前端如何做监控体系(埋点 → 上报 → 分析 → 数据分流)
前端·js or ts
盖丽男2 小时前
彻底搞懂:前端MVVM、后端MVC、DDD极致面向对象的区别与落地真相
前端·mvc
澄江静如练_2 小时前
vue2中使用provide和inject出现的问题
前端·vue
星辰徐哥3 小时前
表单优化:AI驱动HTML5表单的智能验证与提示功能
前端·人工智能·html5
普通网友3 小时前
HTML5新增了哪些重要标签?多多学习也是成长的一部分
前端·学习·html5
2501_906467633 小时前
html5网页中如何实现内网大文件的加密下载?
前端·html·html5·vue上传解决方案·vue断点续传·vue分片上传下载·vue分块上传下载
何何____3 小时前
css变换语法介绍及案例展示
前端·css