Array.prototype.map()详解
今天在地铁上看到这样一个小例子:
- ["1","2","3"].map(parseInt);
相信很多人和我一样,觉得输出的结果是[1,2,3]。no!no!!no!!!正确的答案是[1,NaN,NaN]。当时我百思不得其解,于是到了公司之后就开始查阅资料,终于弄明白了。
我们先来介绍一下map()方法:
概述:
map()方法返回一个由原数组中的每个元素调用一个指定方法后的返回值组成的新数组,它不会改变原来的数组。
语法:
array.map(callback[, thisArg])
参数:
callback
原数组中的元素调用该方法后返回一个新数组。它接收三个参数,分别为 currentValue、index、array。
currentValue
callback的第一个参数,数组中当前被传递的元素。
index
callback的第二个参数,数组中当前被传递的元素的索引。
array
callback的第三个参数,调用map()方法的数组,即原数组。
thisArg
执行callback函数时this指向的对象。
描述
map()方法会给原数组中的每个元素都按顺序调用一次callback函数。callback每次执行后的返回值组合起来形成一个新的数组。callback函数只会在有值的索引上被调用,那些从来没被赋过值或者使用delete删除的索引则不会被调用。
callback函数会被自动传入三个参数:数组元素、数组元素索引、原数组本身。
如果thisArg参数有值,则每次调用callback函数时,this都会指向thisArg参数上的这个对象。如果省略了thisArg参数,或者赋值为null或undefined,则this指向全局对象。
使用map()方法处理数组时,数组元素的范围是在callback函数第一次被调用之前就确定了。在map()方法执行的过程中:原数组中新增加的元素将不会被callback访问到;若已经存在的元素被改变或删除了,则它们传递到callback的值是map()方法遍历到它们那一时刻的值;而被删除的元素将不会被访问到。
总结
通常情况下,map()方法中的callback函数只接受一个参数,就是正在被遍历的数组元素本身。但这并不意味着map只给callback传了一个参数。这个思维惯性可能会让我们犯一个很容易犯的错误。
兼容旧环境
map()方法是在最近的 ECMA-262 标准中新添加的方法;所以一些旧版本的浏览器可能没有实现该方法。在那些没有原生支持 map() 方法的浏览器中,你可以使用下面的 Javascript 代码来实现它。
所使用的算法正是 ECMA-262,第 5 版规定的。假定Object、TypeError和 Array 有他们的原始值。而且callback.call的原始值也是 Function.prototype.call
- // 实现 ECMA-262, Edition 5, 15.4.4.19
- // 参考: http://es5.github.com/#x15.4.4.19
- if (!Array.prototype.map) {
- Array.prototype.map = function(callback, thisArg) {
- var T, A, k;
- if (this == null) {
- throw new TypeError(" this is null or not defined");
- }
- // 1. 将O赋值为调用map方法的数组.
- var O = Object(this);
- // 2.将len赋值为数组O的长度.
- var len = O.length >>> 0;
- // 3.如果callback不是函数,则抛出TypeError异常.
- if (Object.prototype.toString.call(callback) != "[object Function]") {
- throw new TypeError(callback + " is not a function");
- }
- // 4. 如果参数thisArg有值,则将T赋值为thisArg;否则T为undefined.
- if (thisArg) {
- T = thisArg;
- }
- // 5. 创建新数组A,长度为原数组O长度len
- A = new Array(len);
- // 6. 将k赋值为0
- k = 0;
- // 7. 当 k < len 时,执行循环.
- while(k < len) {
- var kValue, mappedValue;
- //遍历O,k为原数组索引
- if (k in O) {
- //kValue为索引k对应的值.
- kValue = O[ k ];
- // 执行callback,this指向T,参数有三个.分别是kValue:值,k:索引,O:原数组.
- mappedValue = callback.call(T, kValue, k, O);
- // 返回值添加到新数组A中.
- A[ k ] = mappedValue;
- }
- // k自增1
- k++;
- }
- // 8. 返回新数组A
- return A;
- };
- }
了解了map()方法,现在我们回过头来看看文章开头提到的那个小例子。
通常使用parseInt时,只需要传递一个参数。但实际上,parseInt可以有两个参数,第二个参数是进制数,可以通过语句“alert(parseInt.length) === 2”来验证。
map()方法在调用callback函数时,会给它传递三个参数:当前正在遍历的元素、元素索引、原数组本身。第三个参数parseInt会忽视,但第二个参数不会,也就是说,parseInt把传过来的索引值当成进制数来使用,而parseInt的第二个参数的范围为2~36(不包含2但包含36),如果省略该参数或者其值为0,则数字将以10为基数来解析;如果小于2或者大于36,parseInt()将返回NaN,所以最终返回了[1,NaN,NaN]。
实际上,这个小例子可以分解成这样理解:
- parseInt("1",0); //基数为0,以十进制来解析,返回1
- parseInt("2",1) //基数为1,小于2,返回NaN
- parseInt("3",2) //基数为2,小于2,返回NaN
注意:在测试parseInt()方法第二个参数范围的过程中发现,当传入的值为1,基数为2时,也会返回1,这点需要注意一下。
我们再来看一下其他小例子:
1、将数组中的单词转换成复数形式。
- function fuzzyPlural(single){
- var result = single.replace(/o/g,"e");
- if(single === "kangaroo"){
- result += "se";
- }
- return result;
- }
- var words = ["foot","goose","moose","kangaroo"];
- console.log(words.map(fuzzyPlural));
- //返回["feet","geese","meese","kangareese"];
2、求数组中每个元素的平方根。
- var numbers = [1, 4, 9];
- var roots = number.map(Math.sqrt);
- //返回[1, 2, 3]
3、在字符串上使用map方法,实现如何在一个String上使用map方法获取字符串中每个字符所对应的ASCII码组成的数组:
- var map = Array.prototype.map;
- var a = map.call("hello world",function(x){ return x.charCodeAt(0);})
- //a的值为[72, 101, 108, 108, 111, 32, 87, 111, 114, 108, 100]
Array.prototype.map()详解的更多相关文章
- Javascript中Array.prototype.map()详解
map 方法会给原数组中的每个元素都按顺序调用一次 callback 函数.callback 每次执行后的返回值组合起来形成一个新数组. callback 函数只会在有值的索引上被调用:那些从来没被赋 ...
- JavaScript的Array.prototype.filter()详解
摘抄与:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Array/filter 概述 ...
- Array.prototype.map()方法详解
Array.prototype.map() 1 语法 const new_array = arr.map(callback[, thisArg]) 2 简单栗子 let arr = [1, 5, 10 ...
- Javascript中prototype属性详解 (存)
Javascript中prototype属性详解 在典型的面向对象的语言中,如java,都存在类(class)的概念,类就是对象的模板,对象就是类的实例.但是在Javascript语言体系中,是不 ...
- 深入理解 Array.prototype.map()
深入理解 Array.prototype.map() map() 方法创建一个新数组,其结果是该数组中的每个元素都调用一个提供的函数后返回的结果. 语法 let new_array = arr.map ...
- Array.prototype.map()
mdn上解释的特别详细 概述 map() 方法返回一个由原数组中的每个元素调用一个指定方法后的返回值组成的新数组. 语法 array.map(callback[, thisArg]) 参数 callb ...
- Sass map详解
作为一个CSS预处理器,Sass正受到越来越多的青睐,诸如Github.Codepen.CSS-Tricks.SitePoint.w3cplus等网站采用Sass组织.管理CSS文件,Sass正在逐渐 ...
- Array.prototype.map()和Array.prototypefilter()
ES5 => 筛选功能 Array.prototypefilter(): 代码: var words = ['spray', 'limit', 'elite', 'exuberant', 'd ...
- ECMAScript5中新增的Array方法实例详解
ECMAScript5标准发布于2009年12月3日,它带来了一些新的,改善现有的Array数组操作的方法.(注意兼容性) 在ES5中,一共有9个Array方法:http://kangax.githu ...
随机推荐
- 【10】令operator=返回一个reference to *this
1.令operator= 返回一个reference to *this,为什么? 这只是一个协议,并无强制性.但是,为了与基本类型的行为保持一致性,强烈建议这么做.设计class 有一个宝典:一旦有疑 ...
- Qualcomm Web Site For Android Development
https://www.codeaurora.org/xwiki/bin/QAEP/release https://support.cdmatech.com/login/ https://chipco ...
- python内存管理
python对象三要素: identity(值):对应于内存的地址,不可修改 type(类型):不可修改 value(值): mutable :可以修改 immutable:不可以修改 引用计数 当引 ...
- eclipse启动不了报错java was started but returned exit code=13
http://zhidao.baidu.com/question/1367307106041927459.html http://zhidao.baidu.com/question/570567914 ...
- 六、Socket之UDP异步传输文件-实现稳定的文件传输
上一篇文章五.Socket之UDP异步传输文件-实现传输中取消传送中,还遗留了一个传输文件最大的问题,就是传输过程中丢包,这样在文件传输过程中就会卡住了,这篇文章就来解决文件传输中的丢包问题,实现稳定 ...
- com.ulitis.www
package com.ulitis.www; import java.io.BufferedReader; import java.io.BufferedWriter; import java.io ...
- java_jdbc_基本连接池
MyDataSource实现封装连接池,MyConnection实现connection类,通过代理模式相互调用 package cn.itcast; import java.sql.*; publi ...
- Oracle建表实例
建表一般来说是个挺简单的事情,但是Oracle的建表语句有很多可选的参数,有些我们可能平时不太用,用的时候又不知道怎么用,这里就写一个较完整的建表的例子: CREATE TABLE banping ...
- Ⅱ.spring的点点滴滴--对象
承接上文 对象的各种实例化 .net篇(环境为vs2012+Spring.Core.dll) 修改原来的PersonDao对象为 public class PersonDao : IPersonDao ...
- swift和oc区别----属性部分(参考官方swift2.1文档)
对于实用过OC的人来说实用swift上手时非常容易的,swift包括了oc的大部分功能,但是swift毕竟是一门新的编程语言,它和OC还是 有很多不同的地方,而且提供了不少新功能,所以本人在读swif ...