视频1 视频21 视频41 视频61 视频文章1 视频文章21 视频文章41 视频文章61 推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37 推荐39 推荐41 推荐43 推荐45 推荐47 推荐49 关键词1 关键词101 关键词201 关键词301 关键词401 关键词501 关键词601 关键词701 关键词801 关键词901 关键词1001 关键词1101 关键词1201 关键词1301 关键词1401 关键词1501 关键词1601 关键词1701 关键词1801 关键词1901 视频扩展1 视频扩展6 视频扩展11 视频扩展16 文章1 文章201 文章401 文章601 文章801 文章1001 资讯1 资讯501 资讯1001 资讯1501 标签1 标签501 标签1001 关键词1 关键词501 关键词1001 关键词1501 专题2001
解析JavaScript循环和作用域之间的关系(附代码)
2020-11-27 19:31:08 责编:小采
文档


我们在工作中经常会用到JavaScript,他与我们的工作有密切的关系。但是JavaScript有一个特点,也许会让开发者头痛, 是其循环和作用域相关的,那接下里就说说他们之间的关系。

举个例子:

const operations = []
for (var i = 0; i < 5; i++) {
 operations.push(() => {
 console.log(i)
 })
}
for (const operation of operations) {
 operation()
}

它基本是循环了5次,将一个函数添加到operations数组里面。这个函数可打印出循环变量索引值i.
运行这些函数后期望的结果应该是:
0 1 2 3 4

但实际发生的是这样的:

5 5 5 5 5

为什么会有这种情况? 因为使用的是var.由于提升了var变量, 上面的代码等同于

var i;
const operations = []
for (i = 0; i < 5; i++) {
 operations.push(() => {
 console.log(i)
 })
}
for (const operation of operations) {
 operation()
}

因此,在for-of循环中, i 依然是可见的, 它等于5,并且每次在函数中涉及到i ,都将使用这个值。
那么我们应该如何做让其变成我们想的这样呢?
最简单的方案是用 let 声明. 在ES2015中介绍到, 它们有很大的帮助,能避免关于使用var声明的一些奇怪问题。
简单的在循环变量时将var 变成 let ,能够很好的运行:

const operations = []
for (let i = 0; i < 5; i++) {
 operations.push(() => {
 console.log(i)
 })
}
for (const operation of operations) {
 operation()
}

这是输出结果:

0 1 2 3 4

这是怎么实现的呢?这是因为每次循环重复的时候,都将重新创造 i ,同时每个函数添加operations数组时,能获取它本身的i。
记住你不能使用 const在这种情况下, 因为这会导致for在第二次循环时, 尝试赋新值报错。
另外一个非常普遍的解决这个问题是使用pre-ES6代码, 同时它被称作即时调用函数表达式 (IIFE).
在这种情况下,你可以包装整个函数,并将i 绑定在它上面。自这种方式,你正在创造一个能立即执行的函数,你从其返回的一个新的函数。因此我们可以稍后执行它。

const operations = []
for (var i = 0; i < 5; i++) {
 operations.push(((j) => {
 return () => console.log(j)
 })(i))
}
for (const operation of operations) {
 operation()
}

下载本文
显示全文
专题