ホームページ >ウェブフロントエンド >jsチュートリアル >JS浮動小数点演算のバグを解決する方法_JavaScriptスキル
37.5*5.5=206.08 (JS がこの結果を計算しました。小数点第 2 位に四捨五入しました)
最初は丸めの問題ではないかと思ったので、JS を直接使用して結果を計算しました: 206.08499999999998
どうやってこれができますか小数点が 1 つしかない 2 つの数値を乗算するときに、どうしてこのような余分な小数点が発生するのでしょうか?
グーグルで調べたところ、これは JavaScript 浮動小数点演算のバグであることがわかりました。
例: 7*0.8 JavaScript は次のように計算します: 5.6000000000000005
オンラインでいくつかの解決策を見つけました。それは、いくつかの浮動小数点演算関数を書き直すか、複数の演算を直接拡張することです。
同じ問題に遭遇した友人の参考のために、これらの方法を以下に抜粋します。
関数 accDiv(arg1,arg2){
var t1=0,t2=0,r1,r2;
try{t1=arg1.toString().split(".")[1]。 length}catch(e){}
try{t2=arg2.toString().split(".")[1].length}catch(e){}
with(Math){
r1=数値(arg1.toString().replace(".",""))
r2=数値(arg2.toString().replace(".",""))
return (r1/ r2)*pow(10,t2-t1);
}
}
//呼び出しをより便利にするために、Div メソッドを Number 型に追加します。
Number.prototype.div = function (arg){
return accDiv(this, arg)
}
//正確な乗算結果を取得するために使用される乗算関数
//説明: JavaScript の乗算結果にはエラーが発生します。これは、2 つの浮動小数点数を乗算する場合にさらに顕著になります。この関数は、より正確な乗算結果を返します。
//呼び出し: accMul(arg1,arg2)
//戻り値: arg1 に arg2 を乗算した正確な結果
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)
}
//呼び出しをより便利にするために、Number 型に mul メソッドを追加します。
Number.prototype.mul = function (arg){
return accMul(arg, this)
}
//正確な加算結果を取得するために使用される加算関数
//説明: JavaScript の加算結果にはエラーが含まれます。これは、2 つの浮動小数点数を加算するときにより顕著になります。この関数は、より正確な加算結果を返します。
//呼び出し: accAdd(arg1,arg2)
//戻り値: arg1 と arg2 の正確な結果
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
}
//呼び出しをより便利にするために、Number 型に add メソッドを追加します。
Number.prototype.add = function (arg){
return accAdd(arg,this)
}
//説明: JavaScript の減算結果にはエラーが含まれます。これは、2 つの浮動小数点数を加算するときにより顕著になります。この関数は、より正確な減算結果を返します。
//呼び出し: accSubtr(arg1,arg2)
//戻り値: arg1 から arg2 を引いた正確な結果
function accSubtr(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);
}
//数値を与えるtype subtr メソッドを追加して、呼び出すのをより便利にします。
Number.prototype.subtr = function (arg){
return accSubtr(arg,this)
}
/ 小数点以下の桁数がわかっている場合は、浮動小数点数を整数に増幅して (最後に対応する倍数で除算して) 演算を実行すると、正しい結果が得られます