为啥javascript小数相减会出现一长串的小数位数?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了为啥javascript小数相减会出现一长串的小数位数?相关的知识,希望对你有一定的参考价值。

例如地址栏输入javascript:document.write(11.3-10.1)
会显示1.200000000000001
它为什么要如此画添足,虽然基本不影响数据的正确性,可是实在是太难看了

javascript:document.write( (11.3-10.1).toFixed(2) )

toFixed()方法不仅仅截去多余的小数位,同时它还根据截取位置的下一个小数位进行四舍五入。例如,对于数值10.739,截取到小数点后的两位数,结果将是10.74。而对于数值10.732,截取到小数点后的两位数,结果将是10.73。

注意,在JavaScript中我们只能截取小数点之后0位~20位的小数。

toFixed()方法仅被高版本的浏览器所支持,所以在使用之前最好先检查一下浏览器是否支持该方法,检查的代码如下所示:

var varNumber = 22.234;

if (varNumber.toFixed)

varNumber = varNumber.toFixed(2);

else //浏览器不支持toFixed()就使用其他方法

var div = Math.pow(10,2);
varNumber = Math.round(varNumber * div) / div;

这样可以解决,但你想问,怎么可能多出这么小数点出来。
为什么会出现如此无法理解的答案?

我Google了一下,发现原来这是JavaScript浮点运算的一个bug。
比如:7*0.8 JavaScript算出来就是:5.6000000000000005
网上找到了一些解决办法,就是重新写了一些浮点运算的函数。
下面就把这些方法摘录下来,以供遇到同样问题的朋友参考:

程序代码
//除法函数,用来得到精确的除法结果
//说明:javascript的除法结果会有误差,在两个浮点数相除的时候会比较明显。这个函数返回较为精确的除法结果。
//调用:accDiv(arg1,arg2)
//返回值:arg1除以arg2的精确结果

function accDiv(arg1,arg2)
var t1=0,t2=0,r1,r2;
tryt1=arg1.toString().split(".")[1].lengthcatch(e)
tryt2=arg2.toString().split(".")[1].lengthcatch(e)
with(Math)
r1=Number(arg1.toString().replace(".",""))
r2=Number(arg2.toString().replace(".",""))
return (r1/r2)*pow(10,t2-t1);


//给Number类型增加一个div方法,调用起来更加方便。
Number.prototype.div = function (arg)
return accDiv(this, arg);

//乘法函数,用来得到精确的乘法结果
//说明:javascript的乘法结果会有误差,在两个浮点数相乘的时候会比较明显。这个函数返回较为精确的乘法结果。
//调用:accMul(arg1,arg2)
//返回值:arg1乘以arg2的精确结果
function accMul(arg1,arg2)

var m=0,s1=arg1.toString(),s2=arg2.toString();
trym+=s1.split(".")[1].lengthcatch(e)
trym+=s2.split(".")[1].lengthcatch(e)
return Number(s1.replace(".",""))*Number(s2.replace(".",""))/Math.pow(10,m)

//给Number类型增加一个mul方法,调用起来更加方便。
Number.prototype.mul = function (arg)
return accMul(arg, this);

//加法函数,用来得到精确的加法结果
//说明:javascript的加法结果会有误差,在两个浮点数相加的时候会比较明显。这个函数返回较为精确的加法结果。
//调用:accAdd(arg1,arg2)
//返回值:arg1加上arg2的精确结果
function accAdd(arg1,arg2)
var r1,r2,m;
tryr1=arg1.toString().split(".")[1].lengthcatch(e)r1=0
tryr2=arg2.toString().split(".")[1].lengthcatch(e)r2=0
m=Math.pow(10,Math.max(r1,r2))
return (arg1*m+arg2*m)/m

//给Number类型增加一个add方法,调用起来更加方便。
Number.prototype.add = function (arg)
return accAdd(arg,this);


在你要用的地方包含这些函数,然后调用它来计算就可以了。
比如你要计算:7*0.8 ,则改成 (7).mul(8)
其它运算类似,就可以得到比较精确的结果。

--------------------------------------------------------------------------
以上是在网上一个JS牛人的博客上转载的,不过上面只提及了加法、乘法和除法的解决办法。
这个时候可能很多人就会想,有了加法,减法还不容易?我就是差点让这个想法给害苦了。
其他的就不多说了,帖出减法的代码:

function Subtr(arg1,arg2)
var r1,r2,m,n;
tryr1=arg1.toString().split(".")[1].lengthcatch(e)r1=0
tryr2=arg2.toString().split(".")[1].lengthcatch(e)r2=0
m=Math.pow(10,Math.max(r1,r2));
//last modify by deeka
//动态控制精度长度
n=(r1>=r2)?r1:r2;
return ((arg1*m-arg2*m)/m).toFixed(n);
参考技术A javascript:document.write(paresFloat(11.3-10.1));

这个和数据结构有关系
整数型自动转换成正型计算
小数型直接转成double型计算
这是在内存中运算的时候必须这样
你该知道计算机只认识0和1吧
具体的就是浮点精准度的问题
float 精确到小数点后7位
double 精确到小数点后15位

这在往下说就多了 反正你知道知道这些就OK了 具体的你去学学数据结构和程序运行就知道了本回答被提问者采纳
参考技术B 楼上说的是相减之后的结果会出现一长串的小数是吧
1 parseFloat对字符串进行解析
arseFloat(11.3-10.1).toFixed(1)
2 用toFixed()

以上是关于为啥javascript小数相减会出现一长串的小数位数?的主要内容,如果未能解决你的问题,请参考以下文章

str.Format(_T("%lf"),n1),转换的字符串会出现小数点后的一长串零,怎么去掉?

javascript 整数加减会有精度丢失吗

题目1021:统计字符------------------------主要注意输入的方法,如何让一长串的字符都接收到字符串中

BigDecimal与double

为啥有一长串没有任何底片的 PolygonVertexIndex(在一个 Fbx 文件中)?

为啥javascript接受小数作为整数