JavaScript 的 作用域

Posted 森海书生

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JavaScript 的 作用域相关的知识,希望对你有一定的参考价值。

在看了几本书之后的一些理解和自己的想法。
 
作用域,变量的作用范围
 
在ES6之前
变量的声明
 
  只有var可以声明变量属于某个作用域,并且,也只有全局作用域和函数作用域。  (没有var声明的变量,属于全局作用域,在全局作用域里声明的变量,函数会成为全局的属性)
  所有的变量,不是全局作用域的,就是函数作用域的;
  如果用var 声明变量,并且是在函数中,那么这个变量就属于这个函数,否则,属于全局变量。
 
变量,函数的提升
 
在任何一个作用域中,都存在提升;
 
     对于一个声明,JS引擎进行处理和执行。
 
     所谓的处理,引擎首先会进行全局的扫描,遇到变量的声明(var 声明的变量)就会记录,遇到函数声明(function 关键字开头)也会进行记录,直到全局扫描完毕。然后,引擎开始从头执行,对变量进行修改,对函数进行调用。
     上边所谓的记录,就是提升行为。(如果学过C语言,就知道,函数,变量都要先定义,在使用,否则会报错,但是,js提升,可以理解为,不管你定义在哪里,都会被提升到使用的前面,也就是可以把使用写在定义前面)
 
     引擎会把声明的变量,函数声明记录到全局的作用域,记录有哪些变量存在,并对变量进行初始化赋值,undefined;为什么是记录到全局,而不是对应作用域,主要是因为,一开始扫描,就是扫描全局作用域,它只扫描一级, 它不会深入扫描,只扫描表面。这个其实很好理解,除了变量的提升,还有函数声明的提升,遇到function 关键字,js引擎只是简单的将其提升到最顶层,在全局作用域中定义的函数,它作用域就相当于二级,js引擎是不会在这个时候去扫描函数作用域的。
 
     对于函数作用域的提升,发生在函数被调用的之后,执行之前,同样会进行表面的扫描,记录附属于这个作用域的变量,函数。
 
 
     如果在一个作用域中遇到变量和函数名相同,那么,函数的优先级高,这个标识符任然是函数名。
 
ES6 
   对变量的声明多了,var ,  let ,const;
对于var的,没有变化,同ES6之前
  而let声明变量,const声明常量,对于这两个,他们有的是块级作用域,只要是 {   },围成的区域,就是块级作用域。
在块级作用域里,let,const声明的,都是对内可见,对外不可见。并且,都会附着在这个作用域上。并且,let 和const 不存在提升。
  let 和const 不可重复声明同一个标识符。(具体的相关内容参见ES6相关书籍)。
 
 以上的主要是参考你所不知道的javascript上,一本很厉害的书。
欢迎各位前辈指正,谢谢。
 
 
 

以上是关于JavaScript 的 作用域的主要内容,如果未能解决你的问题,请参考以下文章

闭包理解

eval作用域

对作用域和作用域链的理解

JavaScript中高级进阶推荐一个JavaScript进阶深入系列专题系列(涉及原型作用域执行上下文变量对象this闭包按值传递callapplybindnew继承等

在chrome开发者工具中观察函数调用栈作用域链与闭包

JavaScript 开发进阶:理解 JavaScript 作用域和作用域链