【翻译】JavaScript中5个值得被广泛使用的数组方法
在2009年十月ECMAScript 5被定义正式规范以来,一些可以提高工作效率的数组方法被提出。但是,由于ES5糟糕的浏览器支持率,这些方法并未被开发者广泛使用。
“多余的”数组方法
没有人会质疑这些方法的可用性,但是如果针对它们而编写polyfill(关于polyfill,个人理解就是为了使新方法兼容各浏览器而编写的浏览器“插件”或者扩展,可以参考这篇文章)却并不值得。所以,这些新方法被认为是“good-to-have”而不是“must-have”的,或者干脆被称为“多余的”方法。Ouch!
但是前端技术发展到今天,如果你时常关注Github上一些流行的开源JS项目,你会发现这样一种趋势:开发者们越来越倾向于使用原生代码,而不依赖与第三方库。
所以,推进原生语法的广泛使用度已经非常必要了。
5个值得关注的数组方法
下面,我将介绍ES 5中的非常有用的5个数组方法,这5个方法可以提高开发者的工作效率。
1. indexOF
indexOf方法返回某个元素在数组中的索引值,如果数组中不存在此元素则返回-1
举个栗子:检查“orange”在数组中的位置
(1) 不使用indexOf()
var arr = ['apple','orange','pear'],
found = false; for(var i= 0, l = arr.length; i< l; i++){
if(arr[i] === 'orange'){
found = true;
}
} console.log("found:",found);
(2) 使用indexOf()
var arr = ['apple','orange','pear'];
console.log("found:", arr.indexOf("orange") != -1);
代码简洁了很多吧!
现在我有了新的需求:我想知道数组中所有符合给定条件的元素。怎么办?
2. filter
filter()方法创建一个指定数组中符合给定条件的所有元素组成的新数组
举个栗子:找出数组中name为“orange”的所有元素
(1) 不用filter()
var arr = [
{"name":"apple", "count": 2},
{"name":"orange", "count": 5},
{"name":"pear", "count": 3},
{"name":"orange", "count": 16},
]; var newArr = []; for(var i= 0, l = arr.length; i< l; i++){
if(arr[i].name === "orange" ){
newArr.push(arr[i]);
}
} console.log("Filter results:",newArr);
(2) 使用filter()
var arr = [
{"name":"apple", "count": 2},
{"name":"orange", "count": 5},
{"name":"pear", "count": 3},
{"name":"orange", "count": 16},
]; var newArr = arr.filter(function(item){
return item.name === "orange";
}); console.log("Filter results:",newArr);
代码简洁了很多。
3. foreach()
foreach()方法为数组中的每个元素执行一次给定的方法
举个栗子:for循环和foreach()方法对比
function test(){
var arr = [1,2,3,4,5,6,7,8];
// Uses the usual "for" loop to iterate
for(var i= 0, l = arr.length; i< l; i++){
console.log(arr[i]);
}
console.log("========================");
//Uses forEach to iterate
arr.forEach(function(item,index){
console.log(item);
});
}
foreach()方法是for循环的一种升级写法。本人建议如果可以选择,应该尽量使用foreach()方法。
使用for循环时存在一个容易被忽视的问题:在for循环中声明的变量(比如上例中的var i=0)并不是for循环中的局部变量,而是for循环所在作用域内的局部变量。上例中,在for循环中声明的变量 var i = 0 实际上是方法test()作用域内的局部变量,i非常容易被test()作用域内的其他逻辑访问和重写,从而造成一些问题。
实际上,通过jsperf的测试表明,for循环的性能比foreach()要好很多。
但是,我个人始终认为,除非是处理百万级以上的大数据,否则仍然坚持使用foreach()方法。节省毫秒级的时间并不作为提升产品性能的主要考虑因素。
4. map()
对数组的每个元素调用定义的回调函数并返回包含结果的数组
举个栗子:解析一个数组,为数组中每个元素新增一个fullname属性,并返回新数组
(1) 不使用map()
var oldArr = [{first_name:"Colin",last_name:"Toh"},{first_name:"Addy",last_name:"Osmani"},{first_name:"Yehuda",last_name:"Katz"}];
function getNewArr(){
var newArr = [];
for(var i= 0, l = oldArr.length; i< l; i++){
var item = oldArr[i];
item.full_name = [item.first_name,item.last_name].join(" ");
newArr[i] = item;
}
return newArr;
}
console.log(getNewArr());
(2) 使用map()
var oldArr = [{first_name:"Colin",last_name:"Toh"},{first_name:"Addy",last_name:"Osmani"},{first_name:"Yehuda",last_name:"Katz"}];
function getNewArr(){
return oldArr.map(function(item,index){
item.full_name = [item.first_name,item.last_name].join(" ");
return item;
});
}
console.log(getNewArr());
map()方法在处理包含服务器数据交互操作的重型JavaScript应用中非常适用。
5. reduce()
对数组中的所有元素调用指定的回调函数。 该回调函数的返回值为累积结果,并且此返回值在下一次调用该回调函数时作为参数提供
老实说,在使用reduce()之前我斟酌了很久。reduce()的概念对我来说非常抽象,尤其是“累积”这个词。直到在nodeschool中开始学习一系列的JavaScript方法之后才逐渐掌握reduce()的概念。但是我仍没有发现它有多大作用,直到有一天我在重构自己代码的时候才发现,reduce()简直吊炸天!
举个栗子:解析一个数组,并且返回一个包含每个元素在数组中出现次数的对象
(1) 不用reduce()
var arr = ["apple","orange","apple","orange","pear","orange"];
function getWordCnt(){
var obj = {};
for(var i= 0, l = arr.length; i< l; i++){
var item = arr[i];
obj[item] = (obj[item] +1 ) || 1;
}
return obj;
}
console.log(getWordCnt());
(2)使用reduce()
var arr = ["apple","orange","apple","orange","pear","orange"];
function getWordCnt(){
return arr.reduce(function(prev,next){
prev[next] = (prev[next] + 1) || 1;
return prev;
},{});
}
console.log(getWordCnt());
下面我将解释一下对于reduce()的个人理解。
reduce(callback,initialValue) 有两个参数,callback函数和initialValue。其中callback函数包含4个参数:prev,next,index和array。我们一般只需要用到prev和next两个参数。
prev参数代表数组中的第一个元素,next代表数组中的第二个元素。注意:如果initialValue参数被设置,prev则代表initialValue,而next代表数组中的第一个元素。比如上例中。initialValue被设置为空白对象{},则prev为{}。
/*
* Difference between not passing any parameters
* and passing in a additional parameter into `reduce()`
*/ var arr = ["apple","orange"]; function noPassValue(){
return arr.reduce(function(prev,next){
console.log("prev:",prev);
console.log("next:",next); return prev + " " +next;
});
}
function passValue(){
return arr.reduce(function(prev,next){
console.log("prev:",prev);
console.log("next:",next); prev[next] = 1;
return prev;
},{});
} console.log("No Additional parameter:",noPassValue());
console.log("----------------");
console.log("With {} as an additional parameter:",passValue());
上面的代码中,每次迭代都返回一个值,这个值被当做下一次迭代的prev参数传入。
下面的代码会更加清晰的显示reduce()的功能:
var arr = ["apple","orange","apple","pear"];
function getWordCnt(){
return arr.reduce(function(prev,next,index){
console.log("<b>Iteration "+index+"</b>");
console.log("prev:",prev);
console.log("next:",next);
prev[next] = ++prev[next] || 1;
console.log("Passing this to the 'prev' of the next iteration if any:",prev);
console.log("---------------");
return prev;
},{});
}
console.log("<b>Final Object:</b>",getWordCnt());
Demethodizing
以上提到的这些方法虽然是只针对数组的,但是同时可以被节点列表、Jquery对象甚至字符串使用。我们可以通过一种“demethodizing”的技术来扩展这些数组方法。
// Demethodizing the Array method, forEach(), into a generic "each"
var each = Function.prototype.call.bind([].forEach); var nodeList = document.querySelectorAll("p"); each(nodeList,bold); function bold(node){
node.style.fontWeight ="bold";
}
我们通过 Function.prototype.call.bind 将foreach方法扩展为function类的each方法,现在each方法可以被数组以外的对象使用了。
浏览器支持度
根据ECMAScript 5 compatibility table,以上5个方法可以再所有的手机浏览器和几乎所有桌面浏览器中使用(当我说“所有”的时候,IE9以下的浏览器请自觉离开)。
【翻译】JavaScript中5个值得被广泛使用的数组方法的更多相关文章
- JavaScript中判断整字类型最简洁的实现方法
这篇文章主要介绍了JavaScript中判断整字类型最简洁的实现方法,本文给出多个判断整数的方法,最后总结出一个最短.最简洁的实现方法,需要的朋友可以参考下 我们知道JavaScript提供了type ...
- javascript中apply、call和bind的区别及方法详解
文章目录 apply.call apply.call 区别 apply.call实例 数组之间追加 获取数组中的最大值和最小值 验证是否是数组(前提是toString()方法没有被重写过) 类(伪 ...
- Javascript中两种最通用的定义类的方法
在Javascript中,一切都是对象,包括函数.在Javascript中并没有真正的类,不能像C#,PHP等语言中用 class xxx来定义.但Javascript中提供了一种折中的方案:把对象定 ...
- JavaScript中的内置对象-8--3.Math-Math对象的方法-min()- max()- ceil() - floor()- round()- abs(); Math对象的random()方法;
JavaScript内置对象-3.Math(数值) 学习目标 1.掌握Math对象的方法: min() max() ceil() floor() round() abs() Math.min() 语法 ...
- JavaScript中的内置对象-8--2.String-符串对象 方法; 截取方法; 综合应用; 其他方法;
JavaScript内置对象-2String(字符串) 学习目标 1.掌握字符串对象 方法: charAt() charCodeAt() indexOf() lastIndextOf() charAt ...
- JavaScript 中那些关于坐标和距离的属性与方法
一 前言 在前端开发中总会遇到各种各样需要使用或计算坐标和距离的情况,但是这些属性和方法众多,全部熟练地记下来并非是一件易事,大多只能现查,耗费不少时间精力,于是便有了整理记录的想法,即加深了印象,又 ...
- JavaScript 中的常用12种循环遍历(数组或对象)的方法
1.for 循环 let arr = [1,2,3]; for (let i=0; i<arr.length; i++){ console.log(i,arr[i]) } // 0 1 // 1 ...
- JavaScript中的call()和apply()方法,借此实现继承
1.方法定义 call方法: 语法:call([thisObj[,arg1[, arg2[, [,.argN]]]]]) 定义:调用一个对象的一个方法,以另一个对象替换当前对象. 说明: call ...
- javaScript 中的私有,共有,特权属性和方法
function constructor () { var private_v; // 私有属性 var private_f = function () { // 私有方法 // code }; th ...
随机推荐
- sqlserv 配置 CLR
转载地址:http://www.cnblogs.com/Brambling/p/8000911.html //clr 配置 https://docs.microsoft.com/zh-cn/sql ...
- 【转】Android总结篇系列:Activity启动模式(lauchMode)
[转]Android总结篇系列:Activity启动模式(lauchMode) 本来想针对Activity中的启动模式写篇文章的,后来网上发现有人已经总结的相当好了,在此直接引用过来,并加上自己的一些 ...
- jq无法获取ng-repeat元素,如何控制ng-repeat元素显示与隐藏?
之前都是在做微信小程序的页面,最近做一些html页面,页面也没什么效果,就弄了几个点击事件,控制一些元素的显示与隐藏.后面用angular来写这些页面,然后就遇到了问题,就是用ng-repeat生成的 ...
- thinkphp 5 使用oss
简单的tp5中上传到 图片到oss我本地开发环境为:WAMP;php版本:5.6.19TP版本:5.1.13 1.使用composer 安装 composer require aliyuncs/oss ...
- CODEFORCES ROUND #740 ANALYSES BY TEAM:RED & BLACK
A.Alyona and copybooks Problems: 给你一个数n和代价分别为a, b, c.数量不限的1, 2, 3,求将n凑成4的倍数的最小代价 Analysis: cj:取个模随便凑 ...
- 第三周助教工作总结——NWNU李泓毅
本周点评作业数量:第一部分是第一篇博客的剩余部分,给出了共计22份作业的具体分数.第二部分是第二篇博客的已提交部分共计19份作业. 这是我的博客地址:https://www.cnblogs.com/N ...
- 在windows上传一个新的项目到GitHub上
不多说,直接上步骤 1,新建GitHub的账号密码. 2,新建一个项目 点击new repository 3,选择自己项目,填写格式 点击创建 create repository,这时候一个雏形 ...
- Cache高速缓冲存储器
Cache的命中率:命中Cache的次数比总访问次数 平均访问时间:t(Cache)X命中次数+t(未命中)X未命中次数 Cache与主存的映射方式: 直接映射 全相联映射 组相联映射 图片来源:ht ...
- 分析Ajax抓取今日头条街拍美图
spider.py # -*- coding:utf-8 -*- from urllib import urlencode import requests from requests.exceptio ...
- [转] Java 的泛型擦除和运行时泛型信息获取
原文链接 https://my.oschina.net/lifany/blog/875769 前言 现在很多程序员都会在简历中写上精通 Java.但究竟怎样才算是精通 Java 呢?我觉得不仅要熟练掌 ...