let、const

一直在用却容易被遗忘

for 循环

for (let i = 0; i < 3; i++) {
  let i = 'abc';
  console.log(i);
}
// abc
// abc
// abc

输出了3 次abc。这表明for循环设置循环变量的那部分是一个父作用域,而循环体内部是一个单独的子作用域。

暂时性死区

使用let声明变量时,只要变量在还没有声明完成前使用,就会报错。下面代码帮助理解:

var tmp = 123;
if (true) {
  console.log(typeof tmp);//ReferenceError
  tmp = 'abc'; // ReferenceError
  
  let tmp; //TDZ结束
  console.log(tmp); // undefined

  tmp = 123;
  console.log(tmp); // 123
}

块级作用域

块级作用域必须有大括号,如果没有大括号,JavaScript 引擎就认为不存在块级作用域。

// 第一种写法,报错
if (true) let x = 1;

// 第二种写法,不报错
if (true) {
  let x = 1;
}

第一种写法没有大括号,所以不存在块级作用域,而let只能出现在当前作用域的顶层,所以报错。第二种写法有大括号,所以块级作用域成立。

const

const实际上保证的是变量指向的内存地址所保存的数据不得改动
对于简单类型的数据(数值、字符串、布尔值),变量指向的内存地址,保存的数据就是变量的值,因此等同于常量。
复合类型的数据(主要是对象和数组),变量指向的内存地址,保存的只是一个指向实际数据的指针const只能保证指针不变,指针指向的数据就完全不能控制了。
比如你用const声明了一个对象,你可以给它添加属性,也可以更改已有属性的值,这都不会报错,但是让它指向新的地址就会报错。

const foo = {};

// 为 foo 添加一个属性,可以成功
foo.prop = 123;
foo.prop // 123

// 将 foo 指向另一个对象,就会报错
foo = {}; // TypeError: "foo" is read-only

const a = [];
a.push('Hello'); // 可执行
a.length = 0;    // 可执行
a = ['Dave'];    // 报错

冻结对象,不让更改,使用Object.freeze
详细可参考MDN文档

const foo = Object.freeze({});

// 常规模式时,下面一行不起作用(添加新属性不起作用);
// 严格模式时,该行会报错
foo.prop = 123;

//将对象彻底冻结,包括属性也不让修改
var constantize = (obj) => {
  Object.freeze(obj);
  Object.keys(obj).forEach( (key, i) => {
    if ( typeof obj[key] === 'object' ) {
      constantize( obj[key] );
    }
  });
};
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值