javascript中toFixed()四舍五入使用方法详解


Posted in Javascript onSeptember 28, 2018

最近做的项目涉及到金额的计算,有一种方式就是进行四舍五入的规则进行小数点后面的尾数处理,以前一直以为toFixed方法就是四舍五入的,知道一个用户反馈了金额计算的bug我才如梦初醒(亏了一毛钱),才仔细深究了下toFixed这个方法,唉,还是我不够严谨啊,前车之鉴,大家勿走我的老路!

toFixed还不同的浏览器实现,在IE10及以上里面是正常的四舍五入,但是别的浏览器里面就不一样了,它不是正常的四舍五入(等下重点说),比如:

var a = 1.335;
console.log(a.toFixed(2))
// IE   1.34
//chorme 1.33

其他的浏览器我没去一一测试,所以如果大家用了其他浏览器的还是需要去实际测试一下,我这里就说说javascript的toFixed()方法的四舍五入原理:

toFixed它是一个四舍六入五成双的诡异的方法(也叫银行家算法),"四舍六入五成双"含义:对于位数很多的近似数,当有效位数确定后,其后面多余的数字应该舍去,只保留有效数字最末一位,这种修约(舍入)规则是“四舍六入五成双”,也即“4舍6入5凑偶”这里“四”是指≤4 时舍去,"六"是指≥6时进上,"五"指的是根据5后面的数字来定,当5后有数时,舍5入1;当5后无有效数字时,需要分两种情况来讲:①5前为奇数,舍5入1;②5前为偶数,舍5不进。(0是偶数)

但是,经过我的测试发现,在chorme下面(最新版),并没有完全遵守这个规则,尤其是5的后面没有数字的时候,不是这么判断的,如下:

var b = 1.335
b.toFixed(2)
"1.33"
var b = 1.345
b.toFixed(2)
"1.34"
var b = 1.355
b.toFixed(2)
"1.35"
var b = 1.365
b.toFixed(2)
"1.36"
var b = 1.375
b.toFixed(2)
"1.38"
var b = 1.385
b.toFixed(2)
"1.39"

可以发现在chorme下没有完全去遵循这个规律,或许它有自己的算法,但是毕竟它没有遵循通用的银行家算法,所以tofixed这个方法在涉及到金钱计算的业务中还是少用,
最好别用,否则可能会出大问题!
下面再再说说我自己的做法,就是根据精确位数来取小数点后的数,然后判断精确位是大于4还是小于等于4,上代码吧,不说了:
我们的业务是最多精确到分,也就是两位小数,最少就是取整,不留小数

function moneySwitch(money, precision){//precision是需要精确的位数,如百分位就是2
 var result = 0;
 //先进行一个千分位的四舍五入,保证3.0999这种情况在保留一位小数的时候能是对的,这一位可以这么做没什么问题
 var money = parseFloat(money).toFixed(3);
 try{
  var int_part = money.split(".")[0], //小数点前的整数
  point_num = money.split(".")[1],//取小数点后面的小数
  precision_num = point_num[3-precision];
  if(precision_num>4){//五入的情况
   if(precision==1){
    point_num = parseInt(point_num)+10+"";
    if(point_num.length>3){//说明往整数位进1
     int_part = parseInt(int_part)+1+"";
     point_num = point_num[1]+point_num[2];
    }else{
     point_num = point_num[0]+point_num[1];
    }
    result = parseFloat(int_part+"."+point_num);
   }else if(precision==2){
    point_num = parseInt(point_num)+100+"";
    if(point_num.length>3){//说明往整数位进1
     int_part = parseInt(int_part)+1+"";
     point_num = point_num[1];
    }else{
     point_num = point_num[0];
    }
    result = parseFloat(int_part+"."+point_num);
   }else if(precision==3){
    int_part = parseInt(int_part)+1+"";
    point_num = 0;
   }
   result = parseFloat(int_part+"."+point_num);
  }else{//四舍的情况
   if(precision==1){
    point_num = point_num[0]+point_num[1];
   }else if(precision==2){
    point_num = point_num[0];
   }else if(precision==3){
    point_num = 0;
   }
   result = parseFloat(int_part+"."+point_num);
  } 
 }catch(e){
  return parseFloat(money).toFixed(2);//如果过程中有出错就tofixed代替为解决
 }
 return result;
}

补充:

js处理数字保留2位小数,强制保留2位小数不够补上.00

1、保留两位小数 //功能:将浮点数四舍五入,取小数点后2位

2、//制保留2位小数,如:2,会在2后面补上00.即2.00

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Test</title>
<script type="text/javascript" src="js/jq.js"></script>
</head>
<script type="text/javascript"> 
 //保留两位小数 
 //功能:将浮点数四舍五入,取小数点后2位 
 function toDecimal(x) { 
  var f = parseFloat(x); 
  if (isNaN(f)) { 
   return; 
  } 
  f = Math.round(x*100)/100; 
  return f; 
 } 
 
 
 //制保留2位小数,如:2,会在2后面补上00.即2.00 
 function toDecimal2(x) { 
  var f = parseFloat(x); 
  if (isNaN(f)) { 
   return false; 
  } 
  var f = Math.round(x*100)/100; 
  var s = f.toString(); 
  var rs = s.indexOf('.'); 
  if (rs < 0) { 
   rs = s.length; 
   s += '.'; 
  } 
  while (s.length <= rs + 2) { 
   s += '0'; 
  } 
  return s; 
 } 
  
 function fomatFloat(src,pos){  
   return Math.round(src*Math.pow(10, pos))/Math.pow(10, pos);  
 } 
 
 document.write("四舍五入 <br/>")
 document.write("3.14159267保留2位小数:" + toDecimal(3.14159267)+"<br/>"); 
 document.write("3.14159267强制保留2位小数:" + toDecimal2(3.14159267)+"<br/>"); 
 document.write("3.14159267保留2位小数:" + toDecimal(3.14559267)+"<br/>"); 
 document.write("3.14159267强制保留2位小数:" + toDecimal2(3.15159267)+"<br/>"); 
 document.write("3.14159267保留2位小数:" + fomatFloat(3.14559267, 2)+"<br/>"); 
 document.write("3.14159267保留1位小数:" + fomatFloat(3.15159267, 1)+"<br/>"); 
  
 document.write("五舍六入 <br/>")
 document.write("1000.003保留2位小数:" + 1000.003.toFixed(2)+"<br/>"); 
 document.write("1000.08保留1位小数:" + 1000.08.toFixed(1)+"<br/>"); 
 document.write("1000.04保留1位小数:" + 1000.04.toFixed(1)+"<br/>"); 
 document.write("1000.05保留1位小数:" + 1000.05.toFixed(1)+"<br/>"); 
  
 document.write("科学计数 <br/>")
 document.write(3.1415+"科学技术后:"+3.1415.toExponential(2)+"<br/>"); 
 document.write(3.1455+"科学技术后:"+3.1455.toExponential(2)+"<br/>"); 
 document.write(3.1445+"科学技术后:"+3.1445.toExponential(2)+"<br/>"); 
 document.write(3.1465+"科学技术后:"+3.1465.toExponential(2)+"<br/>"); 
 document.write(3.1665+"科学技术后:"+3.1665.toExponential(1)+"<br/>"); 
 document.write("精确到n位,不含n位 <br/>")
 document.write("3.1415精确到小数点第2位" + 3.1415.toPrecision(2)+"<br/>"); 
 document.write("3.1455精确到小数点第3位" + 3.1465.toPrecision(3)+"<br/>"); 
 document.write("3.1445精确到小数点第2位" + 3.1415.toPrecision(2)+"<br/>"); 
 document.write("3.1465精确到小数点第2位" + 3.1455.toPrecision(2)+"<br/>"); 
 document.write("3.166592679287精确到小数点第5位" + 3.141592679287.toPrecision(5)+"<br/>"); 
</script> 
<body>
<input type="text" id="Score" />
</body>
</html>

这篇关于toFixed()的文章就介绍到这了,希望大家以后多多支持三水点靠木。

Javascript 相关文章推荐
js判断undefined类型,undefined,null, 的区别详细解析
Dec 16 Javascript
jquery判断元素的子元素是否存在的示例代码
Feb 04 Javascript
js动态创建上传表单通过iframe模拟Ajax实现无刷新
Feb 20 Javascript
详解JavaScript表单验证(E-mail 验证)
Mar 31 Javascript
domReady的实现案例
Nov 23 Javascript
Web 开发中Ajax的Session 超时处理方法
Jan 19 Javascript
jQuery中的deferred使用方法
Mar 27 jQuery
浅谈vue的踩坑路
Aug 31 Javascript
详解Vue中watch的高级用法
May 02 Javascript
js运算符的一些特殊用法
Jul 29 Javascript
Echart折线图手柄触发事件示例详解
Dec 16 Javascript
JS script脚本中async和defer区别详解
Jun 24 Javascript
对vue中v-if的常见使用方法详解
Sep 28 #Javascript
总结javascript三元运算符知识点
Sep 28 #Javascript
2种在vue项目中使用百度地图的简单方法
Sep 28 #Javascript
实例分析vue循环列表动态数据的处理方法
Sep 28 #Javascript
js隐式转换的知识实例讲解
Sep 28 #Javascript
vue.js父子组件通信动态绑定的实例
Sep 28 #Javascript
解决webpack+Vue引入iView找不到字体文件的问题
Sep 28 #Javascript
You might like
php上传、管理照片示例
2006/10/09 PHP
10条PHP编程习惯助你找工作
2008/09/29 PHP
PHP 数据库树的遍历方法
2009/02/06 PHP
php中修改浏览器的User-Agent来伪装你的浏览器和操作系统
2011/07/29 PHP
用PHP即时捕捉PHP中的错误并发送email通知的实现代码
2013/01/19 PHP
JQuery小知识
2010/10/15 Javascript
Extjs4 Treegrid 使用心得分享(经验篇)
2013/07/01 Javascript
javascript alert乱码的解决方法
2013/11/05 Javascript
JS复制内容到剪切板的实例代码(兼容IE与火狐)
2013/11/19 Javascript
JavaScript插件化开发教程(五)
2015/02/01 Javascript
jQuery中复合选择器简单用法示例
2018/03/31 jQuery
vue基于mint-ui实现城市选择三级联动
2020/06/30 Javascript
详解react native页面间传递数据的几种方式
2018/11/07 Javascript
浅析Vue 防抖与节流的使用
2019/11/14 Javascript
javascript实现倒计时效果
2020/02/17 Javascript
Python显示进度条的方法
2014/09/20 Python
从Python的源码来解析Python下的freeblock
2015/05/11 Python
python图形绘制奥运五环实例讲解
2019/09/14 Python
详解mac python+selenium+Chrome 简单案例
2019/11/08 Python
python hash每次调用结果不同的原因
2019/11/21 Python
Python&amp;&amp;GDAL实现NDVI的计算方式
2020/01/09 Python
Python实现进度条和时间预估的示例代码
2020/06/02 Python
python 统计list中各个元素出现的次数的几种方法
2021/02/20 Python
html5在移动端的屏幕适应问题示例探讨
2014/06/15 HTML / CSS
英国百年闻名的优质健康产品连锁店:Holland & Barrett
2019/12/19 全球购物
个人简历自我鉴定
2013/10/11 职场文书
大专生简历的自我评价
2013/11/26 职场文书
保密协议书范本
2014/04/22 职场文书
三八红旗集体先进事迹材料
2014/05/22 职场文书
乡镇消防安全责任书
2014/07/23 职场文书
商场消防安全责任书
2014/07/29 职场文书
2016新年致辞
2015/08/01 职场文书
2016暑期师德培训心得体会
2016/01/09 职场文书
oracle通过存储过程上传list保存功能
2021/05/12 Oracle
Spring依赖注入多种类型数据的示例代码
2022/03/31 Java/Android
JS高级程序设计之class继承重点详解
2022/07/07 Javascript