【前端冷知识】如何优雅地取数值的整数和小数部分

如何优雅地获取一个数的整数和小数部分? 这个问题看起来简单,实际上也大有学问。 如果有老司机告诉你,用num|0取整是最简单的方式,你能反驳TA,说出这其中隐藏的坑吗

在处理数值的时候,获取浮点数的整数和小数部分,是一种常见的操作,在JavaScript中有许多方法可以达到目的,但也正因为方法众多,所以哪种方法更好,也值得我们仔细研究一番。

如果有想学习编程的初学者,可来我们的前端直播授课群的哦:571671034里面免费送整套系统的前端教程!

取整数

parseInt比较常用来取整数部分,在一些项目中经常能看到:

letnum =3.75;

console . log ( parseInt ( num ) ) ; // 3

num = - 3.75 ;

console . log ( parseInt ( num ) ) ; // -3

用parseInt取整数,结果是没问题的,但是如果严格来说,其实parseInt并不是设计用来取整数的。

:point_right|type_1_2: 知识点 : parseInt(string, radix) 这个方法是一个将字符串转换为整数的方法,它有两个参数,第一个参数表示要转换的字符串, 如果参数不是一个字符串,则将其转换为字符串 。第二个参数是基数即进制,默认为10。

所以实际上 parseInt(3.75) 这个代码,会先将3.75转为字符串 "3.75" ,然后再将它 parseInt 成为3。

所以用parseInt方法取整数,有两个不好的地方,一是parseInt这个函数名,看起来就是将字符串转整数的,用在这里不是很适合,另一个是转字符串有点多此一举,而且肯定会带来性能开销,所以使用parseInt虽然方便,但不是最好的办法。

那么,有经验的同学,会想到用Math的方法来取整,相关的有3个方法,分别是Math.ceil、Math.round和Math.floor。

其中Math.round是四舍五入的,Math.ceil是向上取整,Math.floor是向下取整。

要达到parseInt的结果,我们需要判断数值的符号,如果是负数,要使用Math.ceil,如果是正数,则使用Math.floor:

functiontrunc(num){

if ( num >= 0 ) return Math . floor ( num ) ;

return Math . ceil ( num ) ;

}

console . log ( trunc ( 3.75 ) ) ; // 3

console . log ( trunc ( - 3.75 ) ) ; // -3

使用Math.round和Math.ceil实现trunc方法,要比使用parseInt的性能好,因为省去了转字符串。我们可以用jsperf测一下:

结果如下图:

看到使用Math.floor+Math.ceil明显要快。

实际上,在ES2015之后,还提供了原生的 Math.trunc ,我们可以更方便地使用Math.trunc,不用自己使用Math.floor和Math.ceil去实现了:

console.log(Math.trunc(3.75));// 3

console . log ( Math . trunc ( - 3.75 ) ) ; // -3

tricky

如果看一些库的代码,你可能会看到这样的取整方式:

letnum =3.75;

console . log ( num | 0 ) ; // 3

num = - num ;

console . log ( num | 0 ) ; // -3

这是一种利用位或“|”操作来取整的手段,老司机经常用,我以前也用。

这是为什么能达到我们的效果呢,具体可以看ECMA-262文档

对位操作的处理中,第5、6步,会把操作数转为Int32,所以我们就可以利用这个特点来使用“|”操作符了。

不过这么做也是有缺陷的,你发现问题了吗?

:point_right|type_1_2: 冷知识 :因为bitwise操作将操作数转为Int32,所以它不能处理超过32位的数值取整,而JavaScript有效整数的范围是53位。

constnum=17179869184.89;

console . log ( num | 0 ) ; // 0

console . log ( Math . trunc ( num ) ) ; // 17179869184

那么用“|”有什么好处呢?如果考虑js文件大小,那么 a|0 与其他方式比较,是最短的方式,所以如果要考虑压缩代码的大小,且明确知道数值范围不会超过32位整数的时候,可以考虑使用这个技巧。

取小数

取了整数部分,接下来取小数部分就很简单了:

functionfract(num){

return num - Math . trunc ( num ) ;

}

console . log ( fract ( 3.75 ) ) ; // 0.75

console . log ( fract ( - 3.75 ) ) ; // -0.75

上面的代码思路就是先用 Math.trunc(num) 取整,然后再与原数相减,就得到了小数部分。

但是,我们还有更加简单的办法:

:point_right|type_1_2: 知识点 :JavaScript的取模运算%并不限于整数运算,可以对浮点数取模。

所以,直接将原数对1取模,即可获得小数部分!

console.log(3.75%1);// 0.75

console . log ( - 3.75 % 1 ) ; // -0.75

这是最简单的取小数的方式,然后反过来,还可以倒推出另一种实现trunc取整的方式:

functiontrunc(num){

return num - num % 1 ;

}

扩展

取小数部分,可以用来实现周期函数,比如实现匀速的js周期动画:

#progress_bar {

display : inline-block ;

width : 0 px ;

height : 20 px ;

background : red ;

}

functionrun(el, duration){

const startTime = Date . now ( ) ;

function update ( ) {

let p = ( Date . now ( ) - startTime ) / duration ;

p % = 1 ;

el . style . width = ` ${ 300  * p } px` ;

requestAnimationFrame ( update ) ;

}

update ( ) ;

}

const bar = document . getElementById ( 'progress_bar' ) ;

run ( bar , 3000 ) ;

如果我们的周期函数要考虑负数那一半区间,其实fract的方式要修改一下:

functionfract(num){

return num - Math . floor ( num ) ;

}

这个方式才是正确的周期,它和之前的实现区别是负数区间返回的值不同,前者负数返回的小数部分为负数,这个实现中,如果num是正数,返回num的小数部分,如果num是负数,返回1.0 + num的负数小数部分,这样就保证返回值始终在0.0~1.0的区间内。

functionfract(num){

return num - Math . floor ( num ) ;

}

console . log ( - 3.75 % 1 ) ; // -0.75

console . log ( fract ( num ) ) ; // 0.25

好了,关于取整和取小数的讨论就到这里。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,692评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,482评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,995评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,223评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,245评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,208评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,091评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,929评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,346评论 1 311
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,570评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,739评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,437评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,037评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,677评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,833评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,760评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,647评论 2 354

推荐阅读更多精彩内容

  • js简介 Js是一种基于事件和对象驱动的解释性、松散性的语言。 一切皆对象 javascript 布兰登艾奇 ...
    塔库纳玛哈哈阅读 1,204评论 0 2
  • 01javascript语法规范 <!DOCTYPE html> javascript语法...
    ouyangqinbin阅读 1,308评论 0 0
  • 简述JavaScript起源起源于美国的Netscape公司,原名为LiveScript,后改为JavaScrip...
    3ab670b99521阅读 3,001评论 0 0
  • 字串符操作与数学函数 字符串 1.字符串的创建 A.使用string()构造函数: 在js里只要用双引号或单引号引...
    一只横行霸道的喵阅读 251评论 0 0
  • 第一次画动物,毛发就画了整幅画的三分之二的时间。还请简书里的大神们多多指点,指出不足啊
    想念刺身阅读 376评论 6 8