zoukankan      html  css  js  c++  java
  • js 预编译

    js 运行代码的时候分为几个步骤:语法分析 ==》预编译  ==》解释执行

    语法解析:通篇扫描代码,查看语法是否出错

    解释执行读一行 - 解释一行 - 执行一行

    预编译执行的操作:

    // 假设之前并没有定义a
    console.log(a);

    打印结果:Uncaught ReferenceError: a is not defined

    这个大家应该都知道:在变量未定义时就对变量进行访问就会报错(typeof 列外)

    再看一个例子:

    1 // 假设之前并没有定义变量a
    2 console.log(a);
    3 
    4 var a = 123;
    5 
    6 console.log(a);

    看上面代码会打印什么?会不会报错?

    打印结果:

    undefined
    123

    为什么这个没有报错:因为在预编译的时候对变量进行了提升即变量提升。 定义的变量的声明(var a ;)被提到了代码块的最前面,变量的赋值操作(a = 123)没有变化。

    所以被编译后就相当于

    1 // 假设之前没有定义变量a
    2 var a;
    3 
    4 console.log(a);
    5 
    6 a = 123;
    7 
    8 console.log(a);

    看下面这个例子:

     1 // 假设之前没有定义test
     2 console.log(test);
     3 
     4 var test = 123; 
     5 
     6 function test () {
     7 
     8 }
     9 
    10 console.log(test);

    打印结果:

    function test () {}
    123

    为什么第二行的输出的不是undefined 了?执行步骤是什么样子的?

    因为函数声明在编译的时候也会进行提升,叫做函数提升

    执行步骤:

    1、变量提升:var test; 提升到最前面。此时test = undefined;

    2、函数提升:function test () {}; 提升到最前面并覆盖var test; 的声明。所以第二行打印function test () {}; 此时test = function test () {}

    3、进行赋值操作 test = 123; 此时test = 123;

     1 function test (test) {
     2   console.log(test);  // function test(){var a = 789}
     3    var test = 123;
     4    function test () {
     5        var a = 789;
     6     };
     7    console.log(test) ;  // 123     
     8 }
     9 
    10 test(456);

    Why?

    变量提升和函数提升是如何实现的?

    预编译执行步骤

    (a) 函数内

     1、在代码定义之后执行之前生成活动对象AO(Activation Object ) 空对象{}

     2、查找函数体内的形参和变量,定义为AO的属性,赋值为undefined;

     3、将实参跟形参统一,即把实参赋值给AO[形参]

     4、查找函数里的函数声明,将函数声明赋值给AO 即 AO[函数名] = 函数

    (b) 全局

     1、在代码定义之后执行之前生成全局环境对象GO(Global variable Object ) 空对象{}

     2、查找变量,定义为GO的属性,赋值为undefined;

     3、查找函数声明,将函数声明赋值给GO 即 GO[函数名] = 函数;

    以上面的例子分析预编译过程:

    function test (test) {
       console.log(test);  // function test(){var a = 789}
        var test = 123;
        function test () {
            var a = 789;
         };
        console.log(test) ;  // 123     
     }
     
    test(456);
    
    var a = 111;

    /** 全局 **/
    // 第一步:创建GO // GO{}; // 第二步:查找变量定义为GO的属性,赋值为undefined;
    // GO{
    // a: undefined
    // };
    // 第三步:将函数声明赋值给GO
    // GO{
    // a: undefined,
    // test: function test(test) {...}
    // }

    /** function test(test){...} 函数内 **/
    // 第一步: 创建AO
    // AO {}
    // 第二步: 查找形参和变量,定义为AO的属性, 赋值为undefined;
    // AO {
    // test: undefined
    // }
    // 第三步: 形参和实参统一
    // AO {
    // test: 456
    // }
    // 第四步:将函数声明给AO
    // AO {
    // test: function test() { var a = 789 }
    // }

    之后就是一步一步的执行代码了

    function test (test) {
       console.log(test);  // function test(){var a = 789}
        var test = 123;
        function test () {
            var a = 789;
         };
        console.log(test) ;  // 123     
     }
     
    test(456);
    
    var a = 111;
    
    // test(456); 进入到function test (test) {...} 内部
    // console.log(test); // 打印 function test(){var a = 789}
    // var test = 123; 由于 test 变量的声明已经提升,所以这句只执行 test = 123; 即AO[test] = 123
    // AO {
    //   test: 123
    // }
    // function test() {var a = 789} 已经提升
    // console.log(test) //打印 123
    // 退出function test (test) {...} 销毁AO
    // var a = 111; 由于 a 变量已经提升,这句只执行赋值 a = 111; 即 GO[a] = 111;
    // GO{
    //   a: 111,
    //   test: function test(test) {...}
    // }
    
    // 当页面销毁时销毁GO
  • 相关阅读:
    HTTPS
    数字签名与数字证书
    oracle 10g 数据库与客户端冲突导致实例创建无监听问题
    javascript正则表达式提取子匹配项
    设计模式的分类
    【2020第一篇】环境问题基础知识
    【致 2020】2020
    【python】写demo 的时候,pycharm 编辑器总是提示"method XX may be static"
    sql 练习题 (二)
    【python项目】json 和dict 的区别
  • 原文地址:https://www.cnblogs.com/linyabin/p/12371740.html
Copyright © 2011-2022 走看看