众所周知,在ES6之前,JavaScript是没有块级作用域的,如下图所示:

  学过其他语言的同学肯定有点诧异,为什么会这样呢?因为js还是不同于其他语言的,在ES5中,只有全局作用域和函数作用域,并没有块作用域,当然我们可以实现块作用域的功能。看下面代码:

  在这段段代码中,我们使用立即执行函数(IIFE)创建了一个局部函数来模仿块级作用域。在ES5时代,JavaScript的作用域只有用全局作用域和局部作用域的说法。到了ES6时代,块级作用域的登场。

  一、关于ES5时代

    1.变量提升

      说到js的变量提升,就不得不说一下js的词法分析。总所周知js代码自上而下执行,但是在js代码执行前,会先进行词法分析。所以js运行要分为词法分析程序执行两个阶段。

      js词法分析主要分为3个步骤:

        1.分析形参:如果函数有形参,则给当前活动对象增加形参属性,默认为undefined。

        2.分析变量声明:如果有类似var a  之类的声明,若没有该属性则增加属性,若已存在则不做操作。默认为undefined。变量的赋值在执行阶段才进行,即执行到该变量的时候才有 a = 11。

        3.分析函数声明:类似 function a(){},若当前活动对象没有该属性则新增否则重写该属性为方法a。

      如图所示,在这段代码中,按照一般的逻辑,第一个console.log会报错为“a is not defined”。

      但是事实上,根据js词法分析的第二步,var a这个声明会被提前到代码的顶部。但是a=1这个赋值却不会,所以这段函数正确的步骤为:

      这就是所谓的变量提升。

    2.函数提升

      在js中,我们常见的常见函数的创建方式有三种——函数构造式(不推荐使用,此处不做分析),函数声明式和函数表达式。下面第一行的代码为函数声明式,第二个为函数表达式。

      

  1. function fn1(){}
  2. var fn2=function(){};

      在以上两种创建方式中,函数表达式的常见方式与普通变量var a=1的创建方式相同,因此它也会受变量提升的影响。而另一种,函数声明式会存在函数提升的情况,并且函数提升比变量提升优先级高!因此分别在创建前打印上述fn1和fn2得到以下结果:

      根据变量提升和函数提升的分析得:

      因此,fn1是作为函数声明被提升到最前面,而fn2先被作为变量创建并提到顶部,然后在相应位置被赋值的。

    3.作用域链

      我们在上面说到js在ES5时代没有块级作用域,只有局部作用域和全局作用域。作用域链用于保证对执行环境有权访问的所有变量和函数的有序访问。看下图,按照一般的思路,b输出为3。这里就用到了作用域链的知识。

      当函数在执行的过程中,先从自己内部找变量。如果找不到,再从创建当前函数所在的作用域去找, 以此往上。因此我们分析调整函数得到下图。在这个函数中给fn赋值为fn1(),即fn1的返回值fn2。在运行fn时,即运行fn2。此时fn2的内部是没用b的,因此我们要去fn2的创建环境中找b=2。所以此处输出为2.

  二、ES6时代

    1.let和const的来临

      首先letconst的作用和var是相同的,但是都是不存在提升,声明的都是块级标识符。大括号内部即形成块级作用域,此时let声明的a在块级作用域外是访问调用不到的

  1. {
  2. let a=1;
  3. }
  4. console.log(a)//报错

      并且let和const都禁止重复声明的:

  1. var a = 30;
  2. var message = 2;
  3. // 这两条都会抛出语法错误
  4. let a = 40;
  5. const message = 1;

      每个const声明的常量必须进行初始化const定义的常量不能修改,但是用const声明的对象可以修改值,即

  1. const a; // 语法错误:常量未初始化
  2. const b = {
  3. name: 'a'
  4. };
  5. b.name = 'b'; // 可以修改
  6.  
  7. // SyntaxError: "person" is read-only
  8. b = {
  9. name: 'c
  10. }

      letconst声明不会var一样提升到作用域顶部,如果在声明之前访问这些变量,会形成所谓的临时死区(Temporal Dead Zone)即使是相对安全的typeof操作符也会触发引用错误。用let来举例(const也一样):

  1. console.log(typeof value);
  2. let value = 1;

      在上面的代码中,let无变量提升的作用,即在let value=1之前的代码出现临时死区(Temporal Dead Zone),即报错value is not defined而不是undefined。

  1. console.log(typeof a);
  2. if(1){
  3. let a=1;
  4. }

      而在上述的代码中,let在块级作用域中,因此在全局作用域中不存在所谓的死区,因此此处打印出undefined。

    2.全局块作用域绑定

      在全局作用域中,var 声明的变量会成为全局对象(浏览器环境中的window)的属性。这意味着var很可能会无意中覆盖一个已经存在的全局变量。

  1.  
  1. var Test=1;
  2. window.Test === Test; // true

      如果在全局作用域中使用let或const,会在全局作用域下创建一个新的绑定,但该绑定不会添加为全局对象的属性。换句话说,用letconst不能覆盖全局变量,而只能遮蔽它。

  1.  
  1. const foo = 1;
  2. window.foo = 2;
  3. console.log(foo); //
  4. console.log(window.foo); //

  在实际开发中,let实际上与我们所用的的var的用法是一样的,直接替换符合逻辑。对于需要些保护的变量,我们要使用const。默认使用const,只有确实需要改变变量的值时使用let

js作用域的相关知识的更多相关文章

  1. js预解析相关知识总结以及一些好玩的面试题

    js预解析的题像在做智力题一样有意思~ 预解析 预解析:在解释这行代码之前发生的事情——变量的声明提前了,函数的声明提前 console.log(num) ——未定义Num,结果是报错 var num ...

  2. 深入理解ES6之——JS类的相关知识

    基本的类声明 类声明以class关键字开始,其后是类的名称:剩余部分的语法看起来像对象字面量中的方法简写,并且在方法之间不需要使用逗号. class Person { //等价于prototype的构 ...

  3. JS的数组相关知识

    创建数组方法一: var a1=new Array(5); console.log(a1.length); console.log(a1); //[] ,数组是空的 var a2=new Array( ...

  4. js鼠标事件相关知识

    1.mousedown->mouseup依次触发后相当于click事件 2.除了mouseenter和mouseleave外,其它的鼠标事件都是冒泡的 3.mouseover和mouseout事 ...

  5. JS作用域相关知识(#精)

    在学习<你不知道的JS>一书中,特将作用域相关知识在此分享一下: #说到作用域,就不得不提到LHS查询和RHS查询: 1)如果查询目的是对变量进行赋值,则使用LHS查询 2)如果查询目的是 ...

  6. JS 作用域与变量提升---JS 学习笔记(三)

    你知道下面的JavaScript代码执行时会输出什么吗? var foo = 1; function bar() { if (!foo) { var foo = 10; } console.log(f ...

  7. 【Python五篇慢慢弹(5)】类的继承案例解析,python相关知识延伸

    类的继承案例解析,python相关知识延伸 作者:白宁超 2016年10月10日22:36:57 摘要:继<快速上手学python>一文之后,笔者又将python官方文档认真学习下.官方给 ...

  8. js作用域问题

    <script type="text/javascript"> alert(i);//Uncaught ReferenceError: i is not defined ...

  9. AJAX跨域调用相关知识-CORS和JSONP(引)

    AJAX跨域调用相关知识-CORS和JSONP 1.什么是跨域 跨域问题产生的原因,是由于浏览器的安全机制,JS只能访问与所在页面同一个域(相同协议.域名.端口)的内容. 但是我们项目开发过程中,经常 ...

随机推荐

  1. Java入门篇(六)——类和对象

    写到这里终于写到了入门篇的最后一个知识点了.类和对象是Java中经常被提到的两个词汇,实际上可以将类看作对象的载体,它定义了对象所具有的功能.Java是面向对象的语言,因此掌握类与对象是学习Java语 ...

  2. c++---天梯赛---大笨钟

    ★题目: ★思路分析: 对可能的情况进行分类处理.在这里我把它们分成了3大类. ①不在敲钟时间 ②在敲钟时间但为整点 ③在敲钟时间且不为整点. 在敲钟时间段内我们可分别对晚8点前后进行分类讨论, 我们 ...

  3. Fontawesome字体使用说明及其常用效果语法

    标签: 字体图标iconfontawesom Font web开发(17) 版权声明:本文为博主原创文章,未经博主允许不得转载. 目录(?)[+] 本文主要介绍如何在我们的站点里引入Footaweso ...

  4. [SinGuLaRiTy] Nescafe 24杯模拟赛

    [SinGularLaRiTy-1044] Copyright (c) SinGuLaRiTy 2017. All Rights Reserved. 小水塘(lagoon) 题目描述 忘川沧月的小水塘 ...

  5. 【开发技术】 java和JSP和JavaScript有什么区别

    JSP全称是:java server page,意思是基于JAVA服务器的网页技术,跟asp,php一样,都是网页制作用的语言 JavaScript:也成为JS,跟JAVA没啥关系,就是赶时髦起个这名 ...

  6. redux学习日志:关于react-redux

    首先先强调一句:一定要多读官方文档,而且要精读,否则你会忽略掉很多东西! 一,Provider 刚开始看的时候,大致浏览了一下,知道了这个组件是能够接收store作为它的属性,然后它里面的子组件就可以 ...

  7. tomcat三种启动不同的启动方式

    Linux下tomcat服务的启动.关闭与错误跟踪,通常通过以下几种方式启动关闭tomcat服务: 切换到tomcat主目录下的bin目录 1. 启动tomcat服务 方式一:直接启动 ./start ...

  8. JavaScript事件高级绑定

    js 进行事件绑定,其中一种不常见的写法是: <div id="father" style="width: 300px; height: 200px; backgr ...

  9. eclipse-java开发实用快捷键

    Expand All:ctrl+小键盘* Collapse All:ctrl+shift+小键盘/

  10. WPF笔记(2.4 Grid)

    第一章已经简单介绍过这个容器,这一节详细介绍.Grid一般是用表格(Grid.Row 和Grid.Column )的,比StackPanel更细致一些,但是,这么玩很麻烦,先横着竖着定义一大堆,然后把 ...