JavaScript中的变量提升

在 ES6 之前,JavaScript 没有块级作用域(一对花括号{}即为一个块级作用域) ,大致分为 全局作用域 和 函数作用域 。变量提升即将变量声明提升到它所在 作用域 的 最开始 的部分。 在 JavaScript 代码运行之前其实是有一个 编译阶段 的。编译之后才是 从上到下 ,一行一行解释执行。变量提升就发生在 编译阶段 ,它把 变量 和 函数 的声明提升至作用域的顶端。(编译阶段的工作之一就是将变量与其作用域进行关联)。我先分开介绍变量提升和函数提升到后面再放到一起比较。 如果想更深入的了解 产生变量提升的原因

注意

同一个变量只会 声明一次 ,其它的会被覆盖掉。

变量提升/函数提升 是提升到 当前作用域 的顶部,如果遇到特殊的 if(){}/try-cache 作用域,同时也会把也会提升到 特殊作用域 的外部。

函数提升 的优先级是高于 变量提升 的优先级,并且 函数声明 和 函数定义 的部分一起被提升。

变量提升

我们直接从代码从最基础的开始

console.log(a);// undefinedvara =2;复制代码

相信这个大家知道,上面代码其实就是

vara;console.log(a);// undefineda =2;复制代码

他会提前声明 a,但是不会给 a 赋值。 但是如下代码会怎么执行呢?

console.log(a);//UncaughtReferenceError:a isnotdefineda =2;复制代码

如果没有通过 var 声明值类型的就不会存在变量提升,而是会报错。

函数提升

声明函数 有两种方式: 一种是 函数表达式 ,另一种是 函数声明 。

函数表达式

console.log(aa)// undefinedvaraa =function(){};/** 代码分解 ***/varaa;console.log(aa);  aa =function(){};复制代码

函数表达式 和 变量 的提升效果基本上是一致的,它会输出 undefined 。

函数声明

它和 函数表达式 是有点不一样的,在没有 {}作用域 时它们表现是一致的。表现一致的例子

console.log(a);// function a () {}functiona(){ };/** 代码分解 ***/functiona(){ };console.log(a);复制代码

那如果 变量提升 和 函数提升 同时存在,谁先谁后呢? 我们根据上面的注意事项 1 和 3 可以得出结果,根据实例来分析一下。 请看下面的例子:

console.log(aa);// function aa () {}varaa ='aaaa';functionaa(){};console.log(aa);// aaaa/** 代码分解 ***/varaa;// 只会声明一次的变量functionaa(){};// 变量别覆盖为 aa 字面量函数console.log(aa);// function aa () {} 输出字面量函数aa ='aaaa';// aa 重新被覆盖为 'aaaa'console.log(aa);// aaaa 输出最后的覆盖值复制代码

其实我们可以通过 chrome 浏览器调试效果大致如下图所示:

到这里就大致知道 变量提升 、 函数提升 它们的大致过程和它们之间的 优先级 。下面我们来说一下它们和 块级作用域 和 函数作用域

的关系。

作用域

在 ES6 出现之后作用域变得很复杂,有太多种了,这里只说和本篇文章相关的几种作用域。我们只看 全局作用域 、 词法作用域 、 块级作用域 、 函数作用域 这四种作用域。 全局作用域 基本上没什么好说的,上面的样例基本上都是 全局作用域 ,这里就不做多的赘述。

词法作用域/函数作用域

词法作用域: 函数在定义它们的作用域里运行,而不是在执行它们的作用域里运行。 我们直接通过一个例子来分析一下:

在有 作用域 时,我们来看一下 函数声明 的表现,还是通过一个实例来分析一下,代码如下:

console.log(aa);// 如果直接输入 会报错 VM1778:1 Uncaught ReferenceError: a is not defined复制代码

下面修改代码来分析在 函数作用域 中 函数声明 的特殊表现。

console.log(aa);// undefinedvaraa ='aaaa';console.log(aa);// aaaafunctiontest(){console.log(aa);// undefinedvaraa ='bbbb';console.log(aa);// bbbb}  test();/** 代码分解 ***/varaa;console.log(aa);// undefinedaa ='aaaa';console.log(aa);// aaaafunctiontest(){varaa;console.log(aa);// undefinedaa ='bbbb';console.log(aa);// bbbb}  test();复制代码

全局声明了一个名字叫做 aa 的变量,它被提升全局域的顶部声明,而在 test 函数中我们又声明了一个变量 aa ,这个变量在当前 函数作用 的顶部声明。在函数的执行的阶段,变量的读取都是就近原则,先从当先的 活动对象 或 作用域 查找,如果没有才会从 全局对象 或 全局作用域 查找。

稍微加大一点难度,修改代码如下:

console.log(aa);// undefinedvaraa ='aaaa';console.log(aa);// aaaafunctiontest(){console.log(aa);// aaaaaa ='bbbb';console.log(aa);// bbbb}  test();/** 代码分解 ***/varaa;console.log(aa);// undefinedaa ='aaaa';console.log(aa);// aaaafunctiontest(){console.log(aa);// aaaaaa ='bbbb';console.log(aa);// bbbb}  test();复制代码

我们把 test函数 内部的 var aa = 'bbbb' 修改为 aa = bbbb ,这样就 不存在变量提升 只是一个简单 变量覆盖赋值 。

块级作用域

在 ES6 中新增了 块级作用域 ,我们可以通过 let/const 来创建 块级作用域 ,只能在当前 块中访问 通过 let/const 声明的变量。 我们简单的了解一下 let 和 块级作用域 ,请看下方的代码:

if(true) {//console.log(aa);//VM439541:1UncaughtSyntaxError:Identifier'aa'has already been declared    let aa ='aaa';  }  console.log(aa);//VM439096:4UncaughtReferenceError:aa isnotdefined复制代码

在 if条件语句 内部通过 let aa = 'aaa' 中的 let 关键字创建了一个 块级作用域 ,所以我们在外面不能访问 aa 变量。

console.log(aa);//VM440010:1UncaughtReferenceError:aa isnotdefinedlet aa ='aaa';复制代码

let 声明的变量同时存在 DTZ(暂时性死区) ,在 let 声明变量之前使用这个变量,会触发 DTZ(暂时性死区) 报错。

letaa ='aaa';letaa ='aaa';  // Uncaught SyntaxError: Identifier'aa'has already been declared复制代码

let 不能多次声明同一个变量,不然会报错。

if判断/try-cache

if(){}/try-cache(){} 它们算一个作用域吗?我们通过下面的例子一步一步的分析它们,我们以 if 为分析样例请看代码:

console.log(aa)// undefinedif(true) {varaa =10;  }console.log(aa);// 10/**代码分析**/vara;console.log(aa);// undefinedif(true) {    aa =10;  }console.log(aa);// 10复制代码

在 变量提升 时 if 是 不存在作用域 的,它的作用域就是全局作用域。那如果是 函数提升 呢? if会存在作用域 吗? 通过下面这个实例我们大概会了解 函数提升 和 if 的关系:

console.log(aa);// undefinedif(true) {console.log(aa);// function aa () {}functionaa(){};console.log(aa);//function aa () {}}/**代码分析**/varaa;console.log(aa);// undefinedif(true) {functionaa(){};console.log(aa);// function aa () {}console.log(aa);//function aa () {}}复制代码

我们通过这个可以看到当前执行的结果和上面所描述的 函数提升 表现并不一致,它只是提升了 aa 的声明,赋值只是发生在 if 内部的,这也是 函数提升 在 if 中特异的表现。再来一个更特异的 if 和 函数提升 。

varaa ='aaaa';if(true) {// **** 执行序号 5console.log(aa);// **** 执行序号 6aa =1;// **** 执行序号 7functionaa(){}// **** 执行序号 8console.log(aa);  }console.log(aa);/**代码分析 执行顺序**/varaa;  aa ='aaaa';if(true) {functionaa(){}console.log(aa);// function aa () {}aa =1;// function aa () {} 再执行一遍console.log(aa);// 1}console.log(aa);// 1 ?这个确定对?复制代码

我们主要观察 if 内部的 aa = 1; function aa () {} 的顺序,在当前代码中 第二个console.log(aa) 会输出一个 1 ,如果我们把 aa = 1; function aa () {} 改为 function aa () {}; aa = 1; 它外部的 console.log(aa) 就会变化,看代码:

varaa ='aaaa';if(true) {// **** 执行序号 1console.log(aa);// function aa () {} **** 执行序号 2functionaa(){}// **** 执行序号 3aa =1;// **** 执行序号 4console.log(aa);// 1}console.log(aa);// function aa () {}/**代码分析 执行顺序**/varaa;  aa ='aaaa';if(true) {functionaa(){}console.log(aa);// function aa () {}// function aa () {} 再执行一遍aa =1;console.log(aa);// 1}console.log(aa);// function aa () {} ?这个确定对?复制代码

如果是按上面分析的代码执行顺序是相同的,但是为什么结果不太相同,这种资料不太好找,我们直接上代码去 chrome 中调试一下代码就一清二楚了,大致调试过程如下:

function aa () {}; aa = 1; 执行过程

执行序号1时: 进入 if 内部执行,在 scope 中会多出来一个 block ,也就是在 作用域链 中会多出来一个 block ,这个作用域中有 aa = function aa () {} 。如下图所示:

这个时候 block 是 function aa() {} 而全局的 window.aa 现在还是 aaaa

执行序号2时: 执行 console.log(aa) ,这个只是一个输出语法并不会改变变量的值,执行效果没有变。

执行序号3时: 执行 function aa() {} , 我们可以看到 block 和 全局作用域 的 aa 变量都改变为 function aa () {} ,如下图所示:

执行序号4时: 它会执行的代码 aa = 1 ,这个时候根据作用域链的规则,就近获取和修改变量。所以 block 内的 aa = 1 ,而全局变量 window.aa = function aa () {} 如下图所示:

aa = 1; function aa () {}; 执行过程

执行序号5时: 进入 if 内部执行,在 scope 中会多出来一个 block ,也就是在 作用域链 中会多出来一个 block ,这个作用域中有 aa = function aa () {} 。如下图所示:

这个时候 block 是 function aa() {} 而全局的 window.aa 现在还是 aaaa

执行序号6时: 执行 console.log(aa) ,这个只是一个输出语法并不会改变变量的值,执行效果没有变。

执行序号7时: 执行 aa = 1 , 我们可以看到 block 作用域的变量 aa 被赋值为了 1 ,而 全局作用域 中的变量 aa 还是 aaaa 。如下图所示:

执行序号8时: 它会执行的代码 function aa() {} ,当前代码执行完成时,我们会发现 全局作用域 中的变量 aa 也被赋值为 1 . 如下图所示:

aa = 1; 执行过程 当没有 function aa () {}; 函数声明时,我们会发现不会产生一个临时的 block 作用域,也不会存在奇特的现象。

综合上面三个实例中我们可以得出以下的结论:

在 if 内部包含了 函数声明 会在内部产生一个 block作用域 ,在不包含时不会产生 block作用域 。

在当前 if 外部存在和 函数声明 相同的 变量名称 时,当执行到 函数声明 时同时会更新外部 函数作用域or全局作用域 中变量的值,只更新当前执行的这一次。

我们再来一个例子来证明我们得到的结论,例子如下:

functiontest(){// debuggervaraa ='aaaa';if(true) {console.log(aa);// 第一个 ƒ aa () {}aa =1;functionaa(){}console.log(aa);// 第二个 1}console.log(aa);// 第三个 1}  test()console.log(aa)// 第四个 VM5607:13 Uncaught ReferenceError: aa is not defined复制代码

第一个 console.log(aa) 会输出 ƒ aa () {} ,因为 函数声明 的提升和赋值都会放到 if 的内部。同时会产生一个 block作用域 。

第二个 console.log(aa) 会输出 if 内部中的 aa = 1 ,因为 a = 1 会把 if 产生的 block作用域 中的变量 aa 修改为了 1 。

第三个 console.log(aa) 会输出 test函数作用域 中的 aa = 1 ,因为在执行 function aa () {} 是都会更新外部变量 aa 的值为 1 ,也就是 test函数作用域 中的 aa = 1 ;

第四个 console.log(aa) 会输出 全局作用域 中的 aa ,因为从来没有声明过全局变量 aa 所以会报错, is not defined 。

来两道题

来两道题加深一下印象。

第一道题

vara =function(){console.log(1);};vara =function(){console.log(2);};vara;console.log(a);a =1;console.log(a);a =2;console.log(a);console.log(typeofa);复制代码

如果只能答出来就没有必要看了。

如果变量提升遇到函数提升,那个优先级更高呢,看下面的代码。

console.log(a);// function a () {console.log(1);}vara =1;functiona(){console.log(1);}console.log(a);// 1复制代码

看上面的代码知道 函数提升 是 高于变量提升 的,因为在 javascript 中函数是一等公民, 并且不会被变量声明覆盖 ,但是会被 变量赋值覆盖 。其实代码如下

vara =function(){console.log(1);};vara;console.log(a);// function a () {console.log(1);}a =1;console.log(a);// 1复制代码

我们再来一个稍微复杂一点的,代码如下:

console.log(a);// function a () {console.log(2);}vara =1;functiona(){console.log(1);}console.log(a);// 1vara =2;functiona(){console.log(2);}console.log(a);// 2console.log(typeofa);// number复制代码

在多次函数提升的会后一个覆盖前一个,然后才是变量提升,其实代码如下:

vara =function(){console.log(1);};vara =function(){console.log(2);};vara;console.log(a);// function a () {console.log(2);}a =1;console.log(a);// 1a =2;console.log(a);// 2console.log(typeofa);// number复制代码

第二道题

第二道题会比第一道题难一点点,代码如下:

console.log(aa);varaa ='aaa';if(true) {console.log(aa);    aa =1;functionaa(){}    aa =2;console.log(aa);  }console.log(aa);复制代码

如果上面的内容看懂了,大概这个题就会感觉很简单,大致过程如下:

第一个 console.log(aa) 会输出 全局作用域 中的 aa 值为 undefined ,因为 var aa = 'aaa' 会产生变量提升,会把 var aa; 放到全局作用域中的顶端,所以会输出 undefined 。

第二个 console.log(aa) 会输出 if 内部中的 aa = ƒ aa () {} , if 内部执行产生 block作用域 ,并且 block作用域 内部的 ƒ aa () {} 被提升到顶部,所以会输出 ƒ aa () {} 。

第三个 console.log(aa) 会输出 block作用域 中的 aa = 2 ,因为在执行 function aa () {} 是都会更新外部变量 aa 的值为 1 ,也就是 全局作用域 中的 aa = 1 ;

第四个 console.log(aa) 会输出 全局作用域 中的 aa ,因为在上一步中我们知道了 全局作用域 中的 aa = 1 ,所以会输出 1 。

undefinedƒaa() {}21复制代码

到此结束JavaScript中的变量提升,如果发现本篇文章没有涉及的变量提升的知识点和错误的地方,请大家多多指正、探讨。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,602评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,442评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,878评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,306评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,330评论 5 373
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,071评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,382评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,006评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,512评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,965评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,094评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,732评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,283评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,286评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,512评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,536评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,828评论 2 345

推荐阅读更多精彩内容