在JavaScript中,咱们常常须要去循环迭代方法操做数组对象等,常见等循环方法有 for
、for in
、for of
、forEach
等。javascript
for循环是最基础常见的一种循环,圆括号中须要三个表达式,由分号分隔,最后面是一个花括号的块语句。前端
for (var i = 0; i <10; i++){ if (i === 5) { continue; //跳出当前循环 } else if (i === 8) { break; //结束循环 } console.log(i); }
continue 语句用来跳出本次循环,但会继续执行后面的循环。
break 语句用来结束循环,后面的循环不会再执行。
⚠️return 并不能用来跳出for循环,return语句只能出如今函数体内,它会终止函数的执行,并返回一个指定的值。vue
你可能会遇到在for循环使用一个异步操做,这也是一个很常见的面试题。在以下场景,你须要将一批id从0到9的用户名数据请求回来,并将id作为key,name为value塞到一个对象里,代码可能会是这样的java
var users = {}; for (var i = 0; i < 10; i++) { ajax.get(`/api/userName/${i}`).then(res => { users[i] = res.name; }); }
最后users对象的数据结果并不是咱们所想的那样,而是{10: '最后一个请求回来的用户名'}
。这是由于异步请求的缘由,因为事件队列机制,for循环会先所有执行完成,而异步请求会在后面的不定时间内完成,而且调用then方法被事件队列排在了后面,而此时在任意一个then方法内i
变量已经在最后一次循环中被递增到等于10,在不停的调用then方法时,users
对象key为10的value会被一直改写直到最后一个请求结束。node
var users = {}; for (let i = 0; i < 10; i++) { ajax.get(`/api/userName/${i}`).then(res => { users[i] = res.name; }); }
将递增变量i使用let
声明便可解决,let 语句声明一个块级做用域的本地变量,花括号里是一个块,每次循环都使用该块级做用域中的变量,能够看做每次循环的块都是相互隔离的,变量只会在该做用域内生效。react
var users = {}; for (var i = 0; i < 10; i++) { (function () { var j = i; ajax.get(`/api/user/${j}`).then(res => { users[j] = res.name; }); }()); }
咱们将异步方法包在一个当即执行函数里面,经过var j
声明的变量去承接在该函数内i
变量的值,因为当即执行函数造成了一个闭包做用域,变量j
在每个做用域内都是单独存在的。webpack
var users = {}; for (var i = 0; i < 10; i++) { (function (value) { ajax.get(`/api/user/${value}`).then(res => { users[value] = res.name; }); }(i)); }
将变量i
做为当即执行函数的参数传递进来,参数也具备各自的做用域,函数参数只在函数内起做用,是局部变量。es6
for...in语句以任意顺序遍历一个对象的可枚举属性,遍历的顺序可能因浏览器实现方式有所不一样。所遍历的内容能够是一个对象、数组、字符串、arguments等。使用Object.defineProperty
方法能够为对象属性定义是否能够枚举。web
在JavaScript中,对象的属性分为可枚举和不可枚举之分,它们是由属性的enumerable
值决定的。可枚举性决定了这个属性可否被for…in查找遍历到。对象的propertyIsEnumerable
方法能够判断此对象是否包含某个属性,而且返回这个属性是否可枚举。
Object, Array, Number等内置的方法和属性都是不可枚举的面试
const obj = {}; Object.defineProperty(obj, 'city', {value: '北京', enumerable: false}); const isEnumerable = obj.propertyIsEnumerable('city'); console.log(obj); // {city: "北京"} console.log(isEnumerable); //false
const obj = {a:1, b:2, c:3}; Object.defineProperty(obj, 'd', {value: 4, enumerable: false}) obj.__proto__ = {name: 'ricky', age: '25'} console.log(obj) console.log('=====for in=======') for (var prop in obj) { console.log(prop, obj[prop]); } console.log('=====Object.keys=======') console.log(Object.keys(obj)) console.log('=====Object.getOwnPropertyNames=======') console.log(Object.getOwnPropertyNames(obj)) console.log('=====Object.values=======') console.log(Object.values(obj)) console.log('=====Object.entries=======') console.log(Object.entries(obj))
输出结果
咱们先使用对象字面量的方式定义量一个obj
,而后使用Object.defineProperty
方法定义key为d
的一个不可枚举属性,而后修改原型链__proto__
,为其赋值了name, age
两个属性。
d
之外的全部可枚举属性,包括其原型链上的属性for in会循环全部可枚举的属性,包括对象原型链上的属性,循环会输出循环对象的key,若是循环的是一个数组则会输出下标索引(index)。
in 运算符测试一个对象其自身和原型链中是否存在该属性。
const obj = {name: 'ricky'}; Object.defineProperty(obj, 'city', {value: '北京', enumerable: false}) obj.__proto__ = {age: '25'} console.log('name' in obj); // true console.log('city' in obj); // true console.log('age' in obj); // true console.log('sex' in obj); // false
for of循环可迭代对象(包括 Array,Map,Set,String,TypedArray,类数组的对象(好比arguments对象、DOM NodeList 对象)、以及Generator生成器对象等)。
const array = [{a: 1}, {b: 2}, {c: 3}]; array.name = 'ricky'; console.log(array) console.log('=====for of=======') for (var prop of array) { console.log(prop); } console.log('=====for in=======') for (var prop in array) { console.log(prop); }
forEach() 方法对数组的每一个元素执行一次提供的函数,其中函数有三个参数,依次为:当前循环项的内容、当前循环的索引、循环的数组。
const array = ['a', 'b', 'c']; array.forEach(function(value, index, data) { console.log(value, index, data); }); // 输出 // a 0 ["a", "b", "c"] // b 1 ["a", "b", "c"] // c 2 ["a", "b", "c"]
map() 方法会依次循环每一项,而且返回结果映射组成一个新的数组。
const array = [1, 2, 3]; const newArray = array.map(function(value, index, data) { return value * 2; }); console.log(newArray); //输出 [2, 4, 6]
使用forEach、map不能中断循环,方法会将每项内容都执行完成才会结束循环。
every循环当返回false时循环即会结束, some方法在循环返回true时结束循环,利用这个特性使用every和some方法均可以跳出循环。
const arr = [1, 2, 3, 4, 5]; arr.every(function(value){ console.log(value); if(value === 3) { //every 循环当返回false时结束循环 return false; } return true //every 循环须要返回true,没有返回值循环也会结束 }); arr.some(function(value){ console.log(value); if(value === 3) { //some 循环当返回true时结束循环 return true; } });