、for 循环
let arr=[1,2,3]; for (let i=0; i<arr.length; i++){ console.log(i,arr[i]) } // 0 1 // 1 2 // 2 3
for 循环是 Js 中最常用的一个循环工具,经常用于数组的循环遍历。
2、for in 循环
let obj={name:'zhou',age:'**'} for(let i in obj){ console.log(i,obj[i]) } // name zhou // age **
for in 循环主要用于遍历普通对象,i 代表对象的 key 值,obj[i] 代表对应的 value,当用它来遍历数组时候,多数情况下也能达到同样的效果,但是你不要这么做,这是有风险的,因为 i 输出为字符串形式,而不是数组需要的数字下标,这意味着在某些情况下,会发生字符串运算,导致数据错误,比如:'52'+1='521' 而不是我们需要的 53。
另外 for in 循环的时候,不仅遍历自身的属性,还会找到 prototype 上去,所以最好在循环体内加一个判断,就用 obj[i].hasOwnProperty(i),这样就避免遍历出太多不需要的属性。
3、while 循环
同样的遍历 cars 数组,先用 for 循环方法
let cars=["BMW","Volvo","Saab","Ford"]; let i=0; for (;cars[i];) { console.log(cars[i]) i++; }; // BMW // Volvo // Saab // Ford
然后是 while 循环方法
cars=["BMW","Volvo","Saab","Ford"]; var i=0; while (cars[i]) { console.log(cars[i] + "<br>") i++; };
我们发现,它们可以实现同样的效果,事实上它们底层的处理是一样的,不过 for 循环可以把定义、条件判断、自增自减操作放到一个条件里执行,代码看起来方便一些,仅此而已。
4、do while 循环
let i=3; do{ console.log(i) i--; } while(i>0) // 3 // 2 // 1
do while 循环是 while 循环的一个变体,它首先执行一次操作,然后才进行条件判断,是 true 的话再继续执行操作,是 false 的话循环结束。
5、Array forEach 循环
let arr=[1,2,3]; arr.forEach(function(i,index){ console.log(i,index) }) // 1 0 // 2 1 // 3 2
forEach循环,循环数组中每一个元素并采取操作, 没有返回值, 可以不用知道数组长度,他有三个参数,只有第一个是必需的,代表当前下标下的 value。
另外请注意,forEach 循环在所有元素调用完毕之前是不能停止的,它没有 break 语句,如果你必须要停止,可以尝试 try catch 语句,就是在要强制退出的时候,抛出一个 error 给 catch 捕捉到,然后在 catch 里面 return,这样就能中止循环了,如果你经常用这个方法,最好自定义一个这样的 forEach 函数在你的库里。
6、Array map()方法
let arr=[1,2,3]; let tt=arr.map(function(i){ console.log(i) return i*2; }) // [2,4,6]
map() 方法返回一个新数组,数组中的元素为原始数组元素调用函数处理后的值。
注意:map 和 forEach 方法都是只能用来遍历数组,不能用来遍历普通对象。
7、Array filter() 方法
let arr=[1,2,3]; let tt=arr.filter(function(i){ return i>1; }) // [2,3]
filter 方法是 Array 对象内置方法,它会返回通过过滤的元素,不改变原来的数组。
8、Array some() 方法
let arr=[1,2,3]; let tt=arr.some(function(i){ return i>1; }) // true
some() 方法用于检测数组中的元素是否满足指定条件(函数提供),返回 boolean 值,不改变原数组。
9、Array every() 方法
let arr=[1,2,3]; let tt=arr.some(function(i){ return i>1; }) // 检测数组中元素是否都大于1 // false
every() 方法用于检测数组所有元素是否都符合指定条件(通过函数提供),返回 boolean 值,不改变原数组。
10、Array reduce()方法
let arr=[1,2,3]; let ad=arr.reduce(function(i,j){ return i+j; }) // 6
reduce() 方法接收一个函数作为累加器,数组中的每个值(从左到右)开始缩减,最终计算为一个值。
11、Array reduceRight()方法
let arr=[1,2,3]; let ad=arr.reduceRight(function(i,j){ return i+j; }) // 6
reduceRight()方法,和 reduce() 功能是一样的,它是从数组的末尾处向前开始计算。
12、for of 循环
for(let i of arr){ console.log(i) } // name // age
for of 循环是 Es6 中新增的语句,用来替代 for in 和 forEach,它允许你遍历 Arrays(数组), Strings(字符串), Maps(映射), Sets(集合)等可迭代(Iterable data)的数据结构,注意它的兼容性。
总结
以上就是我总结的 Js 中常见的循环遍历方法,随着 Es6 标准的兼容性越来越好,我发现很多实现方案慢慢都不再必要了,比如 let、const 取代var 后,在某些情况下的闭包函数也就不存在了。
对前端的技术,架构技术感兴趣的同学关注我的头条号,并在后台私信发送关键字:“前端”即可获取免费的架构师学习资料
知识体系已整理好,欢迎免费领取。还有面试视频分享可以免费获取。关注我,可以获得没有的架构经验哦!!
While语句包括一个循环条件和一段代码块,只要条件为真,就不断循环执行代码块。
while (条件) 语句;
//举例:
var i=0;
while (i < 100) {
console.log('i 当前为:' + i); i=i + 1;
}
do...while循环与while循环类似,唯一的区别就是先运行一次循环体,然后判断循环条件。
do 语句 while (条件);
//举例:
var x=3;
var i=0;
do {
console.log(i); i++;
} while(i < x);
for(var i=0;i<filterarray.length;i++){
alert(filterarray[i]);
}
var obj={a: 1, b: 2, c: 3};
for (var i in obj) {
console.log('键名:', i);
console.log('键值:', obj[i]);
}
// 键名: a // 键值: 1 // 键名: b // 键值: 2
// 其中 obj为循环的对象, i 为对象中的“键名”。如果对象是数组,那么i就是坐标。
注意:fo…in循环一般用于对象的遍历,但是这里有一个坑需要注意:
任何对象都继承了Object对象,或者其它对象,继承的类的属性是默认不可遍历的,for... in循环遍历的时候会跳过,但是这个属性是可以更改为可以遍历的,那么就会造成遍历到不属于自身的属性。
举例来说,对象都继承了toString属性,但是for...in循环不会遍历到这个属性。
var obj={};// toString 属性是存在的obj.toString
// toString() { [native code] }
for (var p in obj) {
console.log(p);
} // 没有任何输出
如果继承的属性是可遍历的,那么就会被for...in循环遍历到。但如果只想遍历自身的属性,使用for...in的时候,应该结合使用hasOwnProperty方法,在循环内部判断一下,某个属性是否为对象自身的属性。否则就可以产生遍历失真的情况。
var person={ name: '老张' };
for (var key in person) {
if (person.hasOwnProperty(key)) {
console.log(key);
}
}// name
此外,for循环遍历json对象有点奇葩:
无规律json数组:
var json=[{dd:'SB',AA:'东东',re1:123}, {cccc:'dd',lk:'1qw'}];
for(var i=0,l=json.length;i<l;i++){
for(var key in json[i]){
alert(key+’:'+json[i][key]);
}
}
为什么要 l=json.length;i<l呢?小伙伴们自己思考下吧!哈哈哈哈……
有规律json数组:
packJson=[
{"name": "nikita", "password": "1111"},
{"name": "tony", "password": "2222"}
];
for (var p in packJson) {//遍历json数组时,这么写p为索引,0,1
alert(packJson[p].name + " " + packJson[p].password);
}
map方法将数组的所有成员依次传入参数函数,然后把每一次的执行结果组成一个新数组返回。
注意:是返回一个新数组,而不会改变原数组。
var numbers=[1, 2, 3];
numbers.map(function (n) {
return n + 1;
});
// [2, 3, 4]
numbers // [1, 2, 3]
map方法接受一个函数作为参数。该函数调用时,map方法向它传入三个参数:当前成员、当前位置和数组本身。
[1, 2, 3].map(function(elem, index, arr) {
return elem * index;
});
// [0, 2, 6]
此外,map()循环还可以接受第二个参数,用来绑定回调函数内部的this变量,将回调函数内部的this对象,指向第二个参数,间接操作这个参数(一般是数组)。
var arr=['a', 'b', 'c'];
[1, 2].map(function (e) {
return this[e];
}, arr)
// ['b', 'c']
上面代码通过map方法的第二个参数,将回调函数内部的this对象,指向arr数组。间接操作了数组arr; forEach同样具有这个功能。
forEach方法与map方法很相似,也是对数组的所有成员依次执行参数函数。但是,forEach方法不返回值,只用来操作数据。也就是说,如果数组遍历的目的是为了得到返回值,那么使用map方法,否则使用forEach方法。forEach的用法与map方法一致,参数是一个函数,该函数同样接受三个参数:当前值、当前位置、整个数组。
function log(element, index, array) {
console.log('[' + index + ']=' + element);
} ;
[2, 5, 9].forEach(log); // [0]=2 // [1]=5 // [2]=9
此外,forEach循环和map循环一样也可以用绑定回调函数内部的this变量,间接操作其它变量(参考上面的map()循环例子)。
filter方法用于过滤数组成员,满足条件的成员组成一个新数组返回。它的参数是一个函数,所有数组成员依次执行该函数,返回结果为true的成员组成一个新数组返回。该方法不会改变原数组。
[1, 2, 3, 4, 5].filter(function (elem) {
return (elem > 3);
}) // [4, 5]
// 上面代码将大于3的数组成员,作为一个新数组返回。
var arr=[0, 1, 'a', false];
arr.filter(Boolean) // [1, "a"]
filter方法的参数函数也可以接受三个参数:当前成员,当前位置和整个数 组。
[1, 2, 3, 4, 5].filter(function (elem, index, arr) {
return index % 2===0;
}); // [1, 3, 5]
此外,filter方法也可以接受第二个参数,用来绑定参数函数内部的this变量。
var obj={ MAX: 3 }; var myFilter=function (item) {
if (item > this.MAX) return true;
};
var arr=[2, 8, 3, 4, 1, 3, 2, 9];
arr.filter(myFilter, obj) // [8, 4, 9]
上面代码中,过滤器myFilter内部有this变量,它可以被filter方法的第二个参数obj绑定,返回大于3的成员。
这两个方法类似“断言”(assert),返回一个布尔值,表示判断数组成员是否符合某种条件。
它们接受一个函数作为参数,所有数组成员依次执行该函数。该函数接受三个参数:当前成员、当前位置和整个数组,然后返回一个布尔值。
some方法是只要一个成员的返回值是true,则整个some方法的返回值就是true,否则返回false。
var arr=[1, 2, 3, 4, 5];
arr.some(function (elem, index, arr) {
return elem >=3;
});
// true
而every方法则相反,所有成员的返回值都是true,整个every方法才返回true,否则返回false。两相比较,some()只要有一个是true,便返回true;而every()只要有一个是false,便返回false.
var arr=[1, 2, 3, 4, 5];
arr.every(function (elem, index, arr) {
return elem >=3;
});
// false
这两个方法在实际开发中,大有可用之处。比如在判定用户是否勾选了不可操作的数据,或者是否勾选了一条可以操作的数据可以使用这两个方法遍历循环数组。
reduce方法和reduceRight方法依次处理数组的每个成员,最终累计为一个值。它们的差别是,reduce是从左到右处理(从第一个成员到最后一个成员),reduceRight则是从右到左(从最后一个成员到第一个成员),其他完全一样。
[1, 2, 3, 4, 5].reduce(function (a, b) {
console.log(a, b);
return a + b;
})
// 1 2
// 3 3
// 6 4
// 10 5
//最后结果:15
reduce方法和reduceRight方法的第一个参数都是一个函数。该函数接受以下四个参数。
累积变量,默认为数组的第一个成员
当前变量,默认为数组的第二个成员
当前位置(从0开始)
原数组
这四个参数之中,只有前两个是必须的,后两个则是可选的。
如果要对累积变量指定初值,可以把它放在reduce方法和reduceRight方法的第二个参数。
[1, 2, 3, 4, 5].reduce(function (a, b) {
return a + b;
}, 10);
// 25
上面的第二个参数相当于设定了默认值,处理空数组时尤其有用,可避免一些空指针异常。
由于这两个方法会遍历数组,所以实际上还可以用来做一些遍历相关的操作。比如,找出字符长度最长的数组成员。
function findLongest(entries) {
return entries.reduce(function (longest, entry) {
return entry.length > longest.length ? entry : longest;
}, '');
}
findLongest(['aaa', 'bb', 'c']) // "aaa"
上面代码中,reduce的参数函数会将字符长度较长的那个数组成员,作为累积值。这导致遍历所有成员之后,累积值就是字符长度最长的那个成员。
Object.keys方法的参数是一个对象,返回一个数组。该数组的成员都是该对象自身的(而不是继承的)所有属性名,且只返回可枚举的属性。
var obj={
p1: 123,
p2: 456
};
Object.keys(obj) // ["p1", "p2"]
Object.getOwnPropertyNames方法与Object.keys类似,也是接受一个对象作为参数,返回一个数组,包含了该对象自身的所有属性名。但它能返回不可枚举的属性。
var a=['Hello', 'World'];
Object.keys(a) // ["0", "1"]
Object.getOwnPropertyNames(a) // ["0", "1", "length"]
上面代码中,数组的length属性是不可枚举的属性,所以只出现在Object.getOwnPropertyNames方法的返回结果中。
由于 JavaScript 没有提供计算对象属性个数的方法,所以可以用这两个方法代替。
var obj={
p1: 123,
p2: 456
};
Object.keys(obj).length // 2
Object.getOwnPropertyNames(obj).length // 2
1.foreach,map,filter循环中途是无法停止的,总是会将所有成员遍历完。
2.他们都可以接受第二个参数,用来绑定回调函数内部的this变量,将回调函数内部的this对象,指向第二个参数,间接操作这个参数(一般是数组)。
forEach循环没有返回值;map,filter循环有返回值。
var f=function (n) {
return 'a'
};
[1, undefined, 2].map(f) // ["a", "a", "a"]
[1, null, 2].map(f) // ["a", "a", "a"]
[1, , 2].map(f) // ["a", , "a"]
上面代码中,map方法不会跳过undefined和null,但是会跳过空位。forEach方法也会跳过数组的空位,这里就不举例了。
some()只要有一个是true,便返回true;而every()只要有一个是false,便返回false.
reduce是从左到右处理(从第一个成员到最后一个成员),reduceRight则是从右到左(从最后一个成员到第一个成员)。
他们都是遍历对象的属性,也是接受一个对象作为参数,返回一个数组,包含了该对象自身的所有属性名。但Object.keys不能返回不可枚举的属性;Object.getOwnPropertyNames能返回不可枚举的属性。
grep()循环能够遍历数组,并筛选符合条件的元素,组成新的数组,并返回。
function(){
var array=[1,2,3,4,5,6,7,8,9];
var filterarray=$.grep(array,function(value){
return value > 5;//筛选出大于5的
});
for(var i=0;i<filterarray.length;i++){
alert(filterarray[i]);
}
for (key in filterarray){
alert(filterarray[key]);
}
}
function(){
var anObject={one:1,two:2,three:3};//对json数组each
$.each(anObject,function(name,value) {
alert(name);
alert(value);
});
var anArray=['one','two','three'];
$.each(anArray,function(n,value){
alert(n);
alert(value);
});
}
inArray()循环能返回参数在数组中对应的坐标。
function(){
var anArray=['one','two','three'];
var index=$.inArray(‘two’,anArray);
alert(index);//返回该值在数组中的键值,返回1
alert(anArray[index]);//value is two
}
$().ready(
function(){
var strings=['0','1','2','3','4','S','6'];
var values=$.map(strings,function(value){
var result=new Number(value);
return isNaN(result) ? null:result;//isNaN:is Not a Number的缩写
});
for (key in values) {
alert(values[key]);
}
});
map循环常用语往数组中添加新元素,第二种写法:
this.detEntityList.map(item=> {
//往比遍历到的对象中添加属性
Object.assign(item, {
sourceType: item.businessType,
})
});
map() 把每个元素通过函数传递到当前匹配集合中,生成包含返回值的新的 jQuery 对象。此用法与原生js的map循环用法一致。
哈哈哈哈,亲爱的同学们,是不是感觉好多啊,其实不要特意去记忆啊,个人建议吧原生for循环,forEach循环,还有Juery的each熟悉就可以啦!如果你再开发者发现这几个循环不好用的话,那就来我这里找找看,这15种循环中,绝对有你有需要的。
了方便例子讲解,现有数组和字面量对象如下
var demoArr=['Javascript', 'Gulp', 'CSS3', 'Grunt', 'jQuery', 'angular'];
var demoObj={
aaa: 'Javascript',
bbb: 'Gulp',
ccc: 'CSS3',
ddd: 'Grunt',
eee: 'jQuery',
fff: 'angular'
};
可以直接看示例,用得太多了,很简单
(function () {
for (var i=0, len=demoArr.length; i < len; i++) {
if (i==2) {
// return; // 函数执行被终止
// break; // 循环被终止
continue; // 循环被跳过
};
console.log('demo1Arr[' + i + ']:' + demo1Arr[i]);
}
})();
关于for循环,有以下几点需要注意
var i=0, len=demo1Arr.length;
for(; i<len; i++) {};
for(var item in arr|obj){} 可以用于遍历数组和对象
(function () {
for (var i in demoArr) {
if (i==2) {
return; // 函数执行被终止
// break; // 循环被终止
// continue; // 循环被跳过
};
console.log('demoArr[' + i + ']:' + demoArr[i]);
}
console.log('-------------');
})();
for in 本质上遍历的是对象,之所以能遍历数组,是因为数组也是一个对象。
var arr=['react', 'vue', 'angular'];
// 等价于
var arr={
0: 'react',
1: 'vue',
2: 'angular'
}
关于for in,有以下几点需要注意:
function res() {
var demoArr=['Javascript', 'Gulp', 'CSS3', 'Grunt', 'jQuery', 'angular'];
for (var item in demoArr) {
if (item==2) {
return;
};
console.log(item, demoArr[item]);
}
console.log('desc', 'function res'); //不会执行
}
因为 for in 的目的是为了遍历对象,因此在遍历时,会同时搜索该对象构造函数上的属性以及原型上的属性,因此 for in 循环相对来说消耗会更大一点。因此,如果有其他更好的选择,则尽量避免考虑使用 for in 循环来遍历数据。
demoArr.forEach(function(arg) {})
参数arg表示数组每一项的元素,实例如下
demoArr.forEach(function (val, index) {
if (e=='CSS3') {
return; // 循环被跳过
// break; // 报错
// continue;// 报错
};
console.log(val, index);
})
具体有以下需要注意的地方
ES5中新增的几个数组方法,forEach, map, filter, reduce等,可以理解为依次对数组的每一个子项进行一个处理(回调函数中的操作),他们是对简单循环的更高一层封装,因此与单纯的循环在本质上有一些不同,所以才会导致 return, continue, break 的不同。
最重要的一点,可以添加第二参数,为一个数组,而且回调函数中的this会指向这个数组。而如果没有第二参数,则this会指向window。
var newArr=[];
demoArr.forEach(function(val, index) {
this.push(val); // 这里的this指向newArr
}, newArr)
虽然在原生中 forEach 循环的局限性很多,但是了解他的必要性在于,很多第三方库会扩展他的方法,使其能够应用在很多地方,比如 angular 的工具方法中,也有 forEach 方法,其使用与原生的基本没有差别,只是没有了局限性,可以在IE下使用,也可以遍历对象
var result=[];
angular.forEach(demoArr, function(val, index) {
this.push(val);
}, result);
函数具体的实现方式如下,不过有一点值得注意的是,当使用 continue时,如果你将 i++ 放在了后面,那么 i++ 的值将一直不会改变,最后陷入死循环。因此使用do/while一定要小心谨慎一点。
// 直接使用while
(function () {
var i=0,
len=demoArr.length;
while (i < len) {
if (i==2) {
// return; // 函数执行被终止
// break; // 循环被终止
// continue; // 循环将被跳过,因为后边的代码无法执行,i的值没有改变,因此循环会一直卡在这里,慎用!!
};
console.log('demoArr[' + i + ']:' + demoArr[i]);
i++;
}
console.log('------------------------');
})();
// do while
(function () {
var i=0,
len=demo3Arr.length;
do {
if (i==2) {
break; // 循环被终止
};
console.log('demo2Arr[' + i + ']:' + demo3Arr[i]);
i++;
} while (i < len);
})();
不建议使用do/while的方式来遍历数组
$.each(demoArr|demoObj, function(e, ele))
可以用来遍历数组和对象,其中e表示索引值或者key值,ele表示value值
$.each(demoArr, function(e, ele) {
console.log(e, ele);
})
输出为
0 "Javascript"
1 "Gulp"
2 "CSS3"
3 "Grunt"
4 "jQuery"
5 "angular"
这里有很多需要注意的地方
console.log(this);
//String {0: "C", 1: "S", 2: "S", 3: "3", length: 4, [[PrimitiveValue]]: "CSS3"}
console.log(this==ele);
// true
$.each(this, function(e, ele) {
console.log(e, ele);
})
// 0 c
// 1 s
// 2 s
// 4 3
为什么 length 和 [[PrimitiveValue]]没有遍历出来?突然灵光一动,在《javascript高级编程》中找到了答案,大概意思就是javascript的内部属性中,将对象数据属性中的 Enumerable 设置为了false
// 查看length的内部属性
console.log(Object.getOwnPropertyDescriptor(this, 'length'));
// Object {value: 4, writable: false, enumerable: false, configurable: false}
(this)` 与this有所不同,不过遍历结果却是一样,你可以在测试代码中打印出来看看
专门用来遍历DOMList
$('.list li').each(function (i, ele) {
console.log(i, ele);
// console.log(this==ele); // true
$(this).html(i);
if ($(this).attr('data-item')=='do') {
$(this).html('data-item: do');
};
})
因为domList并非数组,而是一个对象,只是因为其key值为0,1,2... 而感觉与数组类似,但是直接遍历的结果如下
var domList=document.getElementsByClassName('its');
for(var item in domList) {
console.log(item, ':' + domList[item]);
}
// 0: <li></li>
// 1: <li></li>
// ...
// length: 5
// item: function item() {}
// namedItem: function namedItem() {}
因此我们在使用for in 遍历domList时,需要将domList转换为数组
var res=[].slice.call(domList);
for(var item in res) {}
类似这样的对象还有函数的属性 arguments 对象,当然字符串也是可以遍历的,但是因为字符串其他属性的 enumerable 被设置成了false,因此遍历出来的结果跟数组是一样的,也就不用担心这个问题了.
for of 用于遍历可迭代对象「Iterator」。在 JS 中,数组 Array,字符串 String, Map,Set 等,都是可迭代对象。
对象中包含 Symbol.iterator 属性的,都被称为可迭代对象。
var arr=[1, 2, 3];
arr[Symbol.iterator]
// ? values() { [native code] }
简单案例。
const iterable=['react', 'vue', 'angular'];
for (const value of iterable) {
console.log(value);
}
如果你发现有些人写函数这样搞,不要惊慌,也不要觉得他高大上鸟不起
+function(ROOT, Struct, undefined) {
...
}(window, function() {
function Person() {}
})
()(), !function() {}() +function() {}() 三种函数自执行的方式
学习是一个艰苦的过程,当然如果能把技术学成,最后也一定可以获得高薪工作。掌握一个好的学习方法,跟对一个学习的人非常重要。今后要是大家有啥问题,可以随时来问我,能帮助别人学习解决问题,对于自己也是一个提升的过程。自己整理了一份2020最全面前端学习资料,从最基础的HTML+CSS+JS到HTML5的项目实战的学习资料都有整理web前端学习干货,各种框架都有整理,送给每一位前端小伙伴,想要获取的可以关注我的头条号并在后台私信我:前端,即可免费获取
*请认真填写需求信息,我们会在24小时内与您取得联系。