javascript 计算后结果失精度的问题解决
近日,项目中遇到数值经过数学计算后出现失精度的问题,例如:0.35/10 = 0.034999999999999996。造成这一结果的原因就不再仔细分析了,太繁琐,时间紧也就没仔细去研究。直接说解决方案。
解决方案也是来自网友的分享,大同小异,有扩展Number,给Number增加方法的,也有定义工具方法的。我采用的是扩展Number 的方式。代码如下:
// 解决四维运算,js计算失去精度的问题 //加法 Number.prototype.add = function(arg){ var r1,r2,m; try{r1=this.toString().split(".")[1].length}catch(e){r1=0} try{r2=arg.toString().split(".")[1].length}catch(e){r2=0} m=Math.pow(10,Math.max(r1,r2)) return (this*m+arg*m)/m }; //减法 Number.prototype.sub = function (arg){ return this.add(-arg); }; //乘法 Number.prototype.mul = function (arg){ var m=0,s1=this.toString(),s2=arg.toString(); try{m+=s1.split(".")[1].length}catch(e){} try{m+=s2.split(".")[1].length}catch(e){}
return Number(s1.replace(".",""))*Number(s2.replace(".",""))/Math.pow(10,m) }; //除法 Number.prototype.div = function (arg){ var t1=0,t2=0,t3=0,r1,r2; try{t1=this.toString().split(".")[1].length}catch(e){} try{t2=arg.toString().split(".")[1].length}catch(e){} try{t3=arg.toString().split(".")[0].length}catch(e){} with(Math){ if(this > 1 && (t2 === 0 && arg === pow(10,t3-1))){ return this/arg; }else { r1=Number(this.toString().replace(".","")) r2=Number(arg.toString().replace(".","")) return (r1/r2).mul(pow(10,t2-t1)); } } };
其中,网友分享出来的方法中除法原本只有 else 中的内容,但实际使用过程中发现在处理类似于 3.5/10 的时候仍会出现结果失精度的问题,所以擅自添加了个 if 判断。如有其他更好更全面的解决方式,请广大网友不吝留言告知。
另,附上自定义函数的解决方式:
//除法 function accDiv(arg1, arg2) { var t1 = 0, t2 = 0, t3 = 0, r1, r2; try { t1 = arg1.toString().split(".")[1].length } catch (e) { } try { t2 = arg2.toString().split(".")[1].length t3 = arg2.toString().split(".")[0].length } catch (e) { } r1 = Number(arg1.toString().replace(".", "")) r2 = Number(arg2.toString().replace(".", "")) if (arg1 > 1 && (t2 === 0 && arg2 === Math.pow(10, t3 - 1))) { return arg1 / arg2; } else { r1 = Number(arg1.toString().replace(".", "")) r2 = Number(arg2.toString().replace(".", "")) return accMul((r1 / r2), Math.pow(10, t2 - t1)) } } //乘法 function accMul(arg1, arg2) { var m = 0, s1 = arg1.toString(), s2 = arg2.toString(); try { m += s1.split(".")[1].length } catch (e) { } try { m += s2.split(".")[1].length } catch (e) { } return Number(s1.replace(".", "")) * Number(s2.replace(".", "")) / Math.pow(10, m) } //加法 function accAdd(arg1, arg2) { var r1, r2, m; try { r1 = arg1.toString().split(".")[1].length } catch (e) { r1 = 0 } try { r2 = arg2.toString().split(".")[1].length } catch (e) { r2 = 0 } m = Math.pow(10, Math.max(r1, r2)) return (arg1 * m + arg2 * m) / m } //减法 function Subtr(arg1, arg2) { var r1, r2, m, n; try { r1 = arg1.toString().split(".")[1].length } catch (e) { r1 = 0 } try { r2 = arg2.toString().split(".")[1].length } catch (e) { r2 = 0 } m = Math.pow(10, Math.max(r1, r2)); n = (r1 >= r2) ? r1 : r2; return ((arg1 * m - arg2 * m) / m).toFixed(n); }