最近用淘宝的weex做了个项目,最近稍微闲下来了。正好很久没有接触RN了,所以趁这个机会系统的学习一下ES6的相关知识。
孔子说:没有对比就没有伤害。所以我们要拿ES6和ES5好好对比的学习。这样才能明白es6是多少的好,积极的拥抱她!
1.let
用let声明的变量,只会在let命令所在的代码块内有效。
let的块状作用域
我们来看一个简单的例子:
'use strict';
{
let name = 'cloud';
var age = ;
}
name;//ReferenceError: name is not defined
age;//
我们在let声明变量name的代码块之外去访问这个变量,结果name是没有定义的。如果我们希望在能访问到name应该怎么做呢?
'use strict';
{
let name = 'cloud';
var age = ;
name;//cloud
}
没错,我们在let声明变量的代码块中去访问name变量。结果也正如我们期望的那样,name的值被访问到了。对于高级oo语言而言,他们都是基于块状作用域的。
所以说,let让我们声明的变量也拥有了块状作用域这样一个概念,而不仅仅是函数作用域了。
既然有了块级作用域,那把let用来声明循环里的变量是极好的。
'use strict';
//死循环
for(var i = ; i < ; i++){
console.log(i);
for(var i = ; i < ; i++){
console.log(i);
}
} for(let i = ; i < ; i++){
console.log(i);
for(let i = ; i < ; i++){
console.log(i);
}
}
如果没有let的话,上面代码中的内循环每次都会把var声明的变量i重置为0,所以毫无疑问,这是一个死循环。所以我们必须把内循环中变量i换成变量j,或者其他的名称。但是这也会带来一个副作用,就是循环完成以后的i,被泄露成了全局变量。
'use strict';
for(var i = ; i < ; i++){
i;//0,1,2,3,4
}
i;//
for(let i = ; i < ; i++){
i;//0,1,2,3,4
}
i;//ReferenceError: i is not defined
let声明的变量不存在变量声明提升
'use strict';
name; //undefined
age; //ReferenceError: age is not defined
var name = 'cloud';
let age = ;
用var声明的变量name,虽然我们在name使用后才声明并给其赋值,但是因为变量声明提升的原因,我们看到name已经声明了,只是其值是undefined而已。但是用let声明的age变量,却没有定义。所以说let是不存在变量声明提升的。
let的暂时性死区(temporal dead zone,简称TDZ)
只要是块级作用域中存在let命令,它所在的区域就成为了暂时性死区,使用let命令声明变量之前,该变量都是不可用的。同时该变量不再受到外部的影响。
var name = 'new cloud';
{
//TDZ start
name = 'cloud';
name;//ReferenceError: name is not defined
let name;
//TDZ end
name;//undefined
let name = 'cloud';
name;//cloud
}
如上代码,在let声明name之前,那么这个变量都是不可用的。所以这也带来一个问题,当我们使用typeof这样的反射机制来检测我们的所需的变量是否按照我们设想的那样,typeof不再是一个坚定不移的忠臣了,它也会报错了。
{
typeof name;//undefined
var name = 'cloud';
typeof name;//ReferenceError: name is not defined
let name = 'cloud';
}
let块状作用域内声明的变量,不允许重复声明。
1.
'use strict';
var name = 'cloud';
var name = 'new cloud';
name;//new cloud
2.
'use strict';
{
let name = 'cloud';
var name = 'new cloud';
name;
let name = 'cloud';
var name = 'new cloud';
name;
let name = 'cloud';
let name = 'new cloud';
name;
//SyntaxError: Identifier 'name' has already been declared
}
第二段代码中的所有声明方式,都会报错。name已经被声明。
'use strict';
function setName(name){
let name = name;//SyntaxError: Identifier 'name' has already been declared
{
let name = 'new cloud';
console.log(name);//new cloud
}
console.log(name);//cloud
}
setName('cloud');
同样,在函数中重新声明变量也会报错的。
2.const
const是一个只读的常量。一旦声明,就不可更改。
'use strict';
const PI = 3.14;
PI = 3.1415;//TypeError: Assignment to constant variable
如果用const声明了常量,但是并没有去初始化它。那么,也是会报错的。
'use strict';
const PI;//SyntaxError: Missing initializer in const declaration
同样,const
的作用域与let
命令相同:只在声明所在的块级作用域内有效。const
命令声明的常量也是不提升,同样存在暂时性死区,只能在声明的位置后面使用。const
声明的常量,也与let
一样不可重复声明。
如果我们用const来声明一个引用类型的话。const只能保证指针域不变,但是并不能保证数据域是不可变更的。
'use strict';
const employee = {};
employee['name'] = 'cloud';
employee['age'] = ; employee = {}; //TypeError: "employee " is read-only
employee['name'] = 'new cloud';
console.log(employee.name);//new cloud
如果想把一个对象完成的冻结的话,应该是用Object.freeze()方法。
const employee = Object.freeze({});
employee.name= 'cloud';//TypeError: Can't add property prop, object is not extensible