你不知道的JavaScript(上)草稿
# 前言
本篇为 《你不知道的JavaScript(上)》 的读书笔记。
其实,很多情况下就是因为我们没有熟读规范,导致一些我们认为想当然的用法与实际大相径庭。
该书作者就讲述了这样一些用法,可以避免各个阶段的开发者踩坑。
# ① 作用域和闭包
# 变量提升
foo() //TypeError
var foo = function(){}
/*
foo会被提升到全局,其运行顺序是这样的:
var foo
foo() //TypeError
foo = function(){}
*/
foo() //ReferenceError
let foo = function(){}
if(!obj in window){
var obj = 3
}
console.log(obj)
//obj编译的时候会被提到全局,实际效果是:
var obj
if(!obj in window){
obj = 3
}
console.log(obj)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
# 作用域闭包
想用iife循环输出1~5
for(var i =1;i<=5;i++){
(function(){
setTimeout(function(){console.log(i)},1000)
})()
}
//输出5次6
1
2
3
4
5
6
2
3
4
5
6
这样是不行的,原因在于这个iife中的作用域是空的,setTimeout的执行顺序在循环之后,故又会去拿全局的i
所以需要传入当前的i到iife的作用域,setTimeout执行的时候就去取这个临时变量了
for(var i =1;i<=5;i++){
(function(i){
setTimeout(function(){console.log(i)},1000)
})(i)
}
1
2
3
4
5
2
3
4
5
# 块作用域
想实现es6这样的效果
{
let a = 2;
console.log( a ); // 2
} console.log( a ); // ReferenceError
1
2
3
4
2
3
4
非es6环境下可以这样
try{throw undefined;}
catch(a){
a = 2;
console.log( a ); // 2
}
1
2
3
4
5
2
3
4
5
编辑 (opens new window)
上次更新: 2023/08/23, 09:32:05