zoukankan      html  css  js  c++  java
  • js使用toFixed遇到的问题以及由此引发的小数精度问题

    原文链接: https://www.cnblogs.com/yalong/p/15762637.html

    项目中使用 toFixed 出现的问题:

    一. js报错 Uncaught SyntaxError: Invalid or unexpected token

    如下图所示:

    就是说对 整数 和 字符串 使用toFixed() 会报错

    二. 四舍五入不正确

    1.335.toFixed(2) // 输出 1.33

    四舍五入的问题在谷歌、火狐浏览器下都存在,ie浏览器下正常

    三. 偶尔数字出现特别长的情况,如下图所示:

    问题分析

    问题一其实是toFixed的使用问题

    x.toFixed(n) 方法可把 Number类型的数字x 四舍五入为指定小数位数的数字, n为保留的小数位数,并且返回的结果是字符串类型

    注意这里x 必须为数字Number类型,如果用字符串的话报错

    3.toFixed(2) 也会报错,原因是js引擎在运行的时候,默认将3后面的那个点认为是小数点,所以3.toFixed()也就相当于3.0toFixed(), 所以报错

    解决方法如下:
    1. 多加一个点: 3..toFixed(2) // 输出 3.00
    2. 把数字存一个变量上
    let num = 3
    num.toFixed(2) // 输出 3.00
    
    1. 用括号: (3).toFixed(2) // 输出 3.00

    问题二是浏览器本身toFixed() 的计算有问题

    解决办法可以重写浏览器的toFixed()函数,写法网上很多就不多介绍了

    问题三的本质是js小数的精度问题

    看下面的示例:

    // 加法 =====================
    0.1 + 0.2 = 0.30000000000000004
    0.7 + 0.1 = 0.7999999999999999
    0.2 + 0.4 = 0.6000000000000001
    
    // 减法 =====================
    1.5 - 1.2 = 0.30000000000000004
    0.3 - 0.2 = 0.09999999999999998
     
    // 乘法 =====================
    19.9 * 100 = 1989.9999999999998
    0.8 * 3 = 2.4000000000000004
    
    // 除法 =====================
    0.3 / 0.1 = 2.9999999999999996
    0.69 / 10 = 0.06899999999999999
    
    

    这个问题的原因简单来说就是计算机计算的时候,会先把10进制的小数转为2进制的机器编码,然后使用二进制的编码进行计算,最后再把二进制的结果转为10进制。

    10进制小数转2进制小数的过程如下:

    十进制的小数转换为二进制小数,主要是利用小数部分乘2,取整数部分,直至小数点后为0

    以0.625为例, 如下图所示:

    十进制的 0.625 转为二进制 就是 0.101

    但是有些小数转为二进制的时候,最后一位永远不会是0,然后就变成"无限长度"的了
    看下面例子:

    0.1 + 0.2 = 0.30000000000000004
    

    把0.1 和 0.2 转成二进制如下:

    0.1 -> 0.0001100110011001...(无限)
    0.2 -> 0.0011001100110011...(无限)
    

    IEEE 754 标准的 64 位双精度浮点数的小数部分最多支持 53 位二进制位,所以两者相加之后得到二进制为:

    0.0100110011001100110011001100110011001100110011001100 
    

    因浮点数小数位的限制而截断的二进制数字,再转换为十进制,就成了 0.30000000000000004。
    误差就是这么出现了

    总结

    浏览器的toFixed() 存在的问题如下:

    1. 四舍五入不准确,并且在不同浏览器下也存在差异
    2. 有时会出现小数的精度特别长的情况,当然这个的本质其实是小数的精度问题

    对于以上的问题,可以把toFixed() 方法重写了, 也可以使用别人的轮子,比如: bignumber

    参考链接:
    https://www.cnblogs.com/chyshy/p/14745284.html

    https://www.cnblogs.com/bettermu/p/8532460.html

    https://juejin.cn/post/6844903572979597319

    https://jingyan.baidu.com/article/eb9f7b6dc692e9c79264e878.html

  • 相关阅读:
    第13组_16通信3班_045_OSPFv3作业
    RIPng配置(第十三组)
    基于IPV6的数据包分析(更新拓扑加入了linux主机和抓取133icmp包)(第十三组)
    vmware vsphere powercli 因为在此系统中禁止执行脚本
    vmware virtual machine must be running in order to be migrated
    flashback transaction闪回事务查询
    oracle 闪回功能详解
    linux下修改/dev/shm tmpfs文件系统大小
    vmware虚拟机guest系统重启后获得169.254.X.X的ip解决方法
    一键部署 PPTP server
  • 原文地址:https://www.cnblogs.com/yalong/p/15762637.html
Copyright © 2011-2022 走看看