你不知道的 parseInt |多重惊喜

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: 你不知道的 parseInt |多重惊喜

前言


parseInt 用于解析一个字符串并返回指定基数的十进制整数,使用语法 parseInt(string, radix)radix 表示字符串的基数,值为 2 ~ 36


提起 parseInt ,小包脑海中不由自主会浮现两个东西,其一是 parseInt 的截断用法;其二就是烂大街的 ['1', '2', '3'].map(parseInt) 面试题。尤其是后面这个面试题,最开始简直就是魂牵梦萦的噩梦。但这真的是 parseInt 的全部吗?下面和小包一起来探索 parseInt 的重重惊喜吧。


惊喜一: parseInt (0.0000005) === 5 为 true,大大的疑惑?


小小的脑袋,大大的疑惑,parseInt 转换后的数字为什么会是 5 呐,不应该是 0 吗?(注意传入参数为浮点类型,不是字符串)


我们来看一下 MDNparseInt 语法的解释: parseInt 方法接收两个参数待解析值 string,基数 radixstring 参数如果不是字符串,则会调用其 toString 方法转换为字符串。因此遇到非字符串解析值,parseInt 内部会有两部操作:


Step1: 调用 toString 方法


(0.5).toString(); //  '0.5'
(0.05).toString(); //  '0.05'
(0.005).toString(); //  '0.005'
(0.0005).toString(); //  '0.0005'
(0.00005).toString(); //  '0.00005'
(0.000005).toString(); //  '0.000005'
(0.0000005).toString(); //  '5e-7'
复制代码


注意上面的输出,我们可以发现当数字过小时,toString 输出的结果是科学计数法形式。


Step2: 截断操作


引用一段来自 StackOverflow 的解释:

parseInt 只能将字符串的前导部分解释为整数值;它忽略任何不能被解释为整数的代码单元,并且不会有忽略指示。


因此答案只会返回 5 ,其余的 e-7 被忽略。


注意事项:


如果将要解析的值放入字符串中,那么上面这种惊喜就不会出现了:


parseInt("0.0000005"); // 0
parseInt("0.000000005"); // 0
复制代码


从这个惊喜我们可以收获一个警示:尽可能要避免隐式类型转换。


扩展


数字过小时,调用 toString 方法,会返回科学技术法格式表示;数字过大时,同样也会有类似的情况。


parseInt(999999999999999999999) // 1
// (999999999999999999999).toString()
// '1e+21'
复制代码


但小包在测试过程中发现一些临界值,比较奇怪,有兴趣大家可以去探讨一下。


// 10000000000000000
parseInt(9999999999999999)
复制代码


惊喜二:radix 默认值是 10 吗,何种情况返回值是 NaN?


下面我们就几种情况对这个惊喜进行分析:


2.1 radix 值为 undefined、0 或未指定的


radix 值为 undefined、0 或未指定的,那 JavaScript 会如何处理这种情况:


  1. 如果输入的 string0x0X 开头,那么 radix 会被假定为 16 ,字符串的其他部分按照十六进制来解析。
  2. 如果输入的 string0 开头,ES5 规定使用十进制,但并非所有的浏览器都支持,因此使用 parseInt 时,需要指定 radix
  3. 如果输入的 string 以其他任何值开头,radix 值为 10


我们用 google 浏览器体验一下上述规则。


parseInt("0x16"); // 22
parseInt("0888"); // 888
parseInt("123px"); // 123
复制代码


2.2 radix 值小于 2 或 大于 36


radix 参数的值为 2 ~ 36,当 radix 小于 2 或 大于 36(不包含 0),返回值为 NaN


parseInt("123px", 1); // NaN
parseInt("123px", 38); // NaN
parseInt("123px", -1); // NaN
复制代码


2.3 待转换字符串中,所有的可转换数字都不小于 radix 值


举个栗子: 例如 radix 值为 2 (二进制),而待转换字符串为 '3456',二进制内只有 0、1 是基本算符,因此字符串 '3456' 无法转换成二进制,返回值为 NaN


parseInt("3456", 2); // NaN
parseInt("45px", 3); // NaN
复制代码


2.4 总结


学习完上面几种情况,我们就可以来回答惊喜二的问题了。


  1. radix 默认值是 10 吗?答案: 不是radix 还会考虑 string 参数,如果 string0x(0X) 开头,radix16;若以 0 开头,不同的浏览器有可能有不同的值;其余情况 radix10
  2. 何种情况返回值是 NaN
  • radix 值小于 2 或 大于 36
  • 待转换字符串中,所有的可转换数字都不小于 radix 值(例如 parseInt('345', 2))
  • 第一个非空格字符不能转换为数字(例如 parseInt('a123'))


惊喜三:radix 参数引出的面试题


3.1 ['1', '2', '3'].map(parseInt)


map 函数对数组的每个元素执行回调函数,并返回含有运算结果的新数组。


// val 为数组值
// index 为当前数组索引
// _arr 为当前数组
arr.map((val, index, _arr) => {});
复制代码


因此 ['1', '2', '3'].map(parseInt) 就等同于下面的代码:


[parseInt("1", 0), parseInt("2", 1), parseInt("3", 2)];
复制代码


  1. parseInt("1", 0)符合 2.1radix0,且 string 以字符 1 开始,radix 值为 10,值为 1
  2. parseInt("2", 1)符合 2.2radix 小于 2,返回 NaN
  3. parseInt("3", 2)符合 2.3,待转换字符串中,所有的可转换数字大于 radix 值,返回 NaN


因此该面试题的答案为 [1, NaN, NaN]


3.2 ['10', '10', '10', '10'].map(parseInt)


有了上面的基础,这个题目就好理解了,我们先将这个方法等价转化一下:


[parseInt("10", 0), parseInt("10", 1), parseInt("10", 2), parseInt("10", 3)];
复制代码


相当于将 '10' 分别转化为十进制,一进制,二进制,三进制,因此该面试题的结果是 [10, NaN, 2, 3]


惊喜四:parseInt 的旁支末节


parseInt 方法除了上面提到的惊喜外,还有一些旁支末节,也需要注意一下:


  1. parseInt 允许前导和后置空格,在进行转换前,会首先删除字符串的前导和后置空格。
// 允许前导和后置空格
parseInt("  123 "); // 123
// 中间出现的空格会被当做截断符
parseInt("123 456 "); // 123
复制代码


  1. parseInt 可以理解正负号


parseInt("+123"); // 123
parseInt("-123"); // -123
// 只能理解第一个正负号,第二个非数字字符,因此返回 NaN
parseInt("+-123"); // NaN
复制代码


  1. 使用 parseInt 转换 BigInt 会损失精度


parseIntBigInt 转换为 Number,并在这个过程中失去了精度。这是因为拖尾的非数字值,包括 "n",会被丢弃。


let bigInt = 111111111111111111111111n;
parseInt(bigInt); // 1.1111111111111111e+23
let bigInt2 = 11n;
parseInt(bigInt2); // 11
复制代码


博文仓库


传送门: 战场小包的博客库


别忘了给小包点个 🌟。




相关文章
|
8月前
|
设计模式 中间件 程序员
【C/C++ 奇异递归模板模式 】C++中CRTP模式(Curiously Recurring Template Pattern)的艺术和科学
【C/C++ 奇异递归模板模式 】C++中CRTP模式(Curiously Recurring Template Pattern)的艺术和科学
405 3
|
15天前
|
前端开发 Java 程序员
面试官刁钻提问?轻松应对 break、continue 和 return 的巧妙用法
本次分享的主题是在面试break社招时被问到continue和return的区别与作用,面试官还刁钻的问了一些场景使用的坑点,小伙伴表示不太懂,现场有点慌。今天由我来给大家深入讲讲这三个关键词的区别和作用还会结合一些实战例子,保证你看完后不仅面试游刃有余,临时写代码也更得心应手,我们分为以下四部分。 1.了解背景铺垫的相关知识 2.Break、continue和return的定义 3.使用代码来实现三个关键字的逻辑 4.三个关键字在实践中应注意的坑点
|
5月前
|
Java
"Java排序大揭秘:Comparable与Comparator,究竟有何神秘区别?掌握它们,告别排序难题!"
【8月更文挑战第19天】Java提供Comparable与Comparator两种排序机制。Comparable位于`java.lang`包,定义了`compareTo()`方法以实现类的自然排序;Comparator位于`java.util`包,通过`compare()`方法提供外部定制排序。实现Comparable固定了排序策略,适用于类自带排序逻辑;使用Comparator则可在不改动类的前提下灵活定义多种排序规则,适合多样化的排序需求。选择合适机制可优化排序效率并增强代码灵活性。
32 0
|
8月前
|
存储 Java 程序员
Java程序设计基础——多重循环
Java程序设计基础——多重循环
|
算法 JavaScript
W3Cschool编程实战JS脚本算法挑战:寻找数组中的最大值算法挑战
在右边的大数组中包含了4个小数组,请分别找到每个小数组中的最大值,然后把它们串联起来,形成一个新的数组。
90 0
|
Go 开发者
多重循环应用案例(一)|学习笔记
快速学习多重循环应用案例
多重循环应用案例(一)|学习笔记
|
Go 开发者
多重循环应用案例(二)|学习笔记
快速学习多重循环应用案例
|
JavaScript 前端开发
每日一题:== 和 ===区别,分别在什么情况使用
每日一题:== 和 ===区别,分别在什么情况使用
129 0
竟然有一半的人不知道 for 与 foreach 的区别???
竟然有一半的人不知道 for 与 foreach 的区别???
120 0