JavaScript之map与parseInt的陷阱

问题来源

? 这个问题的来源是学习廖雪峰老师JS教程。问题如下:小明希望利用map()把字符串变成整数,他写的代码很简洁:

‘use strict‘;

var arr = [‘1‘, ‘2‘, ‘3‘];
var r;
r = arr.map(parseInt);
console.log(r);
// [1, NaN, NaN]

为什么不是[1, 2, 3]?这是因为两个两个函数的定义有冲突。下面详解:

map的定义

? 注意到这个问题的原因是参考了这个国外某博客JavaScript可选参数危险。首先,我们了解一下map方法的定义,如下:

var new_array = arr.map(function callback(currentValue[, index[, array]]) {
    // Return element for new_array
}[, thisArg])

其中参数有一个函数callback,这个函数需要三个参数:currentValue(必须):处理的数组中的当前元素;index(可选):当前处理的元素在数组中的索引值;array(可选):调用map的数组。thisArg(可选):对象作为该执行回调时使用,传递给函数,用作 "this" 的值。

如果省略了 thisValue ,"this" 的值为 "undefined"。

? 所以,map函数接收的是两个参数,一个函数,另一个是thisArg。这里我们主要关注的是其中的函数接收三个参数,一个必须,两个可选。问题就是出现在这里,函数接收三个参数。

parseInt的定义

? 这个函数用于解析一个字符串,并返回一个整数。定义如下:

parseInt(string, radix)

参数string(必须):表示的是要被解析的字符串。radix(可选):表示表示要解析的数字的基数。该值介于 2 ~ 36 之间。如果省略该参数或其值为 0,则数字将以 10 为基础来解析。如果它以 “0x” 或 “0X” 开头,将以 16 为基数。如果该参数小于 2 或者大于 36,则 parseInt() 将返回 NaN。第二个可选参数,就是问题的另一来源了。

问题详解

? 了解完两个函数的定义后,我们就知道小明代码的问题了:

var arr = [‘1‘, ‘2‘, ‘3‘];
var r;
r = arr.map(parseInt);

我们通常以为上述代码中 arr.map(parseInt);表示的是

parseInt("1")
parseInt("2")
parseInt("3")

实际上应该是:

parseInt("1", 0, arr)
parseInt("2", 1, arr)
parseInt("3", 2, arr)

其中的arr[‘1‘, ‘2‘, ‘3‘]。我们知道,JavaScript函数通常会忽略额外的参数,并且parseInt只需要两个参数,因此我们不必担心theArray这些调用中参数的影响。但是,第二个参数对parseInt影响很大。parseInt("1", 0)没有问题,0表示的是以二进制为基数解析‘1’。但是后面的parseInt("2", 1)以及parseInt("3", 2)就有问题了。1parseInt中是无效的基数,返回‘NaN’值;2虽然是有意义的基数,但是因为‘3’不是合法的二进制数,所以也返回的是NaN。至此,问题错误已经明了。下面是解决方案。

解决方案

? 解决的方法很简单,就是修饰一些parseInt函数。代码如下:

‘use strict‘;

var arr = [‘1‘, ‘2‘, ‘3‘];
var r;
r = arr.map(function parseInt2(x) {
    return parseInt(x);
});
console.log(r);

或者我们使用Number()方法,这样也会把字符串参数解析成十进制数,并且只需要一个参数。代码如下:

‘use strict‘;

var arr = [‘1‘, ‘2‘, ‘3‘];
var r;
r = arr.map(Number);
console.log(r);

当然,也可以从map下手解决问题。一个可能的方法如下:

Function.prototype.only = function(numberOfArgs) {
   var self = this;     //the original function
   return function() {
      return self.apply(this,[].slice.call(arguments,0,numberOfArgs))
   }
};
arr.map(parseInt.only(1));

以上,就是JavaScript中map与parseInt冲突的问题的来源、解析、解决了。

原文地址:https://www.cnblogs.com/cjvae/p/9782387.html

时间: 2024-10-22 21:27:05

JavaScript之map与parseInt的陷阱的相关文章

JavaScript实现Map、Reduce和Filter

1. [代码][JavaScript]代码     <script type="text/javascript">// 函数式编程:// 描述我们要做什么,而不是我们如何去做.这意味着我们工作在一个更高的抽象层次.函数式编程将导致更精巧.清晰和令人愉快的代码. // 最基础的forEachfunction forEach(array, action) {for (var i = 0; i < array.length; i++) {action(array[i]);}

用JavaScript写map

<script type="text/javascript"> function Map() { this.elements = new Array(); } //获取MAP元素个数 Map.prototype.size = function() { return this.elements.length; }; //判断MAP是否为空 Map.prototype.isEmpty = function() { return (this.elements.length <

JavaScript创建Map对象(转)

JavaScript 里面本身没有map对象,用JavaScript的Array来实现Map的数据结构. Js代码   /* * MAP对象,实现MAP功能 * * 接口: * size()     获取MAP元素个数 * isEmpty()    判断MAP是否为空 * clear()     删除MAP所有元素 * put(key, value)   向MAP中增加元素(key, value) * remove(key)    删除指定KEY的元素,成功返回True,失败返回False * 

javascript实现map的功能(转载)

/* * MAP对象,实现MAP功能 * * 接口: * size() 获取MAP元素个数 * isEmpty() 判断MAP是否为空 * clear() 删除MAP所有元素 * put(key, value) 向MAP中增加元素(key, value) * remove(key) 删除指定KEY的元素,成功返回True,失败返回False * get(key) 获取指定KEY的元素值VALUE,失败返回NULL * element(index) 获取指定索引的元素(使用element.key,

练习:不使用JavaScript内置的parseInt()函数,利用map和reduce操作实现一个string2int()函数

本练习来自廖雪峰JS教程.答案自写. 函数功能描述如下:把一个字符串13579先变成Array--[1, 3, 5, 7, 9],再利用reduce()就可以写出一个把字符串转换为Number的函数. 函数代码如下: 'use strict'; function string2int(s) { var arr = []; // 将字符串转化为数组,但此方法得到的数组的每一个元素仍是字符串 for(var i = 0; i < s.length; ++i) { arr[i] = s[i]; } /

JavaScript Array --&gt;map()、filter()、reduce()、forEach()函数的使用

题目: 1.得到 3000 到 3500 之内工资的人. 2.增加一个年龄的字段,并且计算其年龄. 3.打印出每个人的所在城市 4.计算所有人的工资的总和. 测试数据: function getData() { var arr = [{ id: 1, name: 'ohzri', birth: '1999.09.09', city: '湖北', salary: 9379 }, { id: 2, name: 'rqgfd', birth: '1999.10.28', city: '湖北', sal

当map遇到parseInt

也是一道面试题,估计除了面试题,一般情况下,也不会写出类似的代码了. ['1', '2', '3'].map(parseInt) 这么一道题的返回结果是什么? 如果不用浏览器去验证,乍一看,似乎确实没什么头绪. 我们先看一下map函数的官方解释: map 方法会给原数组中的每个元素都按顺序调用一次callback 函数.callback 每次执行后的返回值(包括 undefined)组合起来形成一个新数组. map本身接受两个参数:callback回调函数和thisArg(可选的)执行 call

C++ std::map::erase用法及其陷阱

1.引入: STL的map中有一个erase方法用来从一个map中删除制定的节点 eg: map<string,string> mapTest; typedef map<string,string>::iterator ITER; ITER iter=mapTest.find(key); mapTest.erase(iter); 像上面这种删除单个节点,map的行为不会出现问题,但是当在一个循环里用的时候,往往会被误用. 2.陷阱 eg: for(ITER iter=mapTest

自定义实现JavaScript的Map对象,修改IE不兼容MAP()的问题

由于IE8及以下版本不支持Map对象,本文为程序猿们提供了有效的解决方法. 本文重写了Map对象,实现了常用的set, get, put, clear, remove, delete, forEach, has, containsKey, isEmpty, size 等方法,使用和声明的方试和正常声明Map对象一样: var map = new Map(); 只需将下面代码拷入<script type="text/javascript"></script>中即可