ECMAScript 变量是松散类型的,意思是变量可以用于保存任何类型的数据。每个变量只不过是一 个用于保存任意值的命名占位符。有 3 个关键字可以声明变量:var
、const
和 let
。
但是三个关键字有什么区别呢?
首先我们来看一下var
,这个是我们最熟悉的。
- var声明的作用域: 使用 var 操作符定义的变量会成为包含它的函数的局部变量。
<div id="app">hello world</div>
在这里,我们在函数内部声明了一个变量age,但是在函数外部却无法使用。造成这样的原因就是var声明的变量是一个局部变量,这个代码里面,var在函数里面声明,那么它的作用域只有函数里面。函数调用完成之后age这个变量就自动销毁了。
最后我们可以变量前面的var来声明一个全局变量,但是我们不推荐这样做,因为这样做在js严格模式下会报错。
- var声明提升
这个声明提升就是把所有的变量声明都自动拉到函数作用域的顶部。我们看一个代码吧。
<div id="app">hello world</div>
在这里面的js函数,age这个变量声明位于输出后面。但是程序依然可以正常运行,没有报错。说明age变量的声明被提升到了函数顶部。- 在var这里,不允许重复声明同一个变量
var name;//这里第一行不会报错
var name;//但是在这里,就会报错,因为第一行已经声明过了
接下来我们看const
let声明和var差不多,但是有一些差别:
- 第一个就是let声明的范围是块作用域,这个块作用域是函数作用域的子集。
块作用域:比如函数里面有一个for循环,然后for循环里面使用let声明了一个变量,那么这个变量就只能在这个for循环里面使用,for循环一结束,这个变量就会被销毁。
在这个代码里面,for循环里面可以正常引用num变量,但是在for循环外面,函数里面却不能正常调用num变量了。这就说明let声明的变量作用域只能是块作用域。
- 在
let
这里,也不可以重复声明同一个变量,原理同var
- 在for循环里面中的let声明,如果在for循环里面使用的不是let进行声明变量,那么就会导致for循环声明的变量渗透到循环体外部,导致外部在没有声明这个变量的同时可以正常引用这个变量。 但是有了let之后,就会将这个变量的作用域限制在for循环里面,就不会渗透到循环体外面,减少了错误的发生率。
第三个就来到了const
:
const
在声明变量的时候,和前面两个有一个很大的不同:在声明变量的同时,必须对其变量进行初始化(赋值)。并且在后面尝试修改const声明变量的值,会导致程序报错。
但是在修改值把报错这里有一个特例:
- const这个限制只适用于它指向的变量的引用,如果const声明的是一个对象,那么对这个对象里面的一些代码进行修改则不会报错。
- 基于const这个不能直接对其声明的变量进行修改的这个特性,我们在for循环里面需要特别注意。就是const不能用来声明for循环里面需求进行迭代的变量,比如
for(const i = 0;i < 5;i++){}
,这里面的i
变量就会报错,因为i
会在这里进行自增。 - 但是,如果你只想用
const
声明一个不会被修改的for
循环变量,那也是可以的。也就是说,每 次迭代只是创建一个新变量。就例如这样:for (const value of [1,2,3,4,5]) { console.log(value); }
这样输出的结果就是1,2,3,4,5.不会报错。