37.5*5.5=206.08 (JS가 계산한 결과, 소수점 이하 두 자리까지 반올림했습니다.)
처음에는 반올림 문제인 줄 알고 JS를 직접 사용해 결과를 계산했습니다. 206.08499999999998
어떻게 이런 일이 가능할까요? 두 숫자에 소수점 하나만 곱할 때 어떻게 그렇게 추가 소수점이 있을 수 있습니까?
Google에서 검색한 결과 이것이 JavaScript 부동 소수점 연산의 버그라는 것을 알았습니다.
예: 7*0.8 JavaScript에서는 다음과 같이 계산합니다. 5.6000000000000005
몇 가지 부동 소수점 연산 함수를 다시 작성하거나 여러 연산을 직접 확장하는 솔루션을 온라인에서 찾았습니다.
같은 문제를 겪는 친구들이 참고할 수 있도록 아래 방법을 발췌했습니다.
function 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);
}
}
//더 편리하게 호출할 수 있도록 Number 유형에 div 메소드를 추가합니다.
Number.prototype.div = 함수(arg){
return accDiv(this, arg)
}
//정확한 곱셈 결과를 얻기 위해 사용되는 곱셈 함수
//설명: JavaScript의 곱셈 결과에는 오류가 있으며, 이는 두 개의 부동 소수점 숫자를 곱할 때 더 분명해집니다. 이 함수는 더 정확한 곱셈 결과를 반환합니다.
//Call: 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 = 함수(arg){
return accMul(arg, this)
}
//정확한 덧셈 결과를 얻기 위해 사용되는 덧셈 함수
//설명: JavaScript의 덧셈 결과에는 오류가 있으며, 이는 두 개의 부동 소수점 숫자를 더할 때 더 분명해집니다. 이 함수는 더 정확한 덧셈 결과를 반환합니다.
//Call: 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 = 함수(arg){
return accAdd(arg,this)
}
//정확한 뺄셈 결과를 얻기 위해 사용되는 뺄셈 함수
//설명: JavaScript의 뺄셈 결과에는 오류가 있으며, 이는 두 개의 부동 소수점 숫자를 더할 때 더 분명해집니다. 이 함수는 보다 정확한 뺄셈 결과를 반환합니다.
//Call: 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 = 함수(arg){
return accSubtr(arg,this)
}
이러한 함수를 사용하려는 곳에 포함시킨 다음 호출하여 계산하세요.
/ 소수 자릿수를 알고 있다면 부동 소수점 수를 정수로 증폭(그리고 마지막으로 해당 배수로 나누기)한 다음 연산을 수행하면 올바른 결과를 얻을 수 있습니다