성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] 그냥 RSS Reader 기능과 약간의 UI 편의성 때문에 사용...
[이종효] 오래된 소프트웨어는 보안 위협이 되기도 합니다. 혹시 어떤 기능...
[정성태] @Keystroke IEEE의 문서를 소개해 주시다니... +_...
[손민수 (Keystroke)] 괜히 듀얼채널 구성할 때 한번에 같은 제품 사라고 하는 것이 아...
[정성태] 전각(Full-width)/반각(Half-width) 기능을 토...
[정성태] Vector에 대한 내용은 없습니다. Vector가 닷넷 BCL...
[orion] 글 읽고 찾아보니 디자인 타임에는 InitializeCompon...
[orion] 연휴 전에 재현 프로젝트 올리자 생각해 놓고 여의치 않아서 못 ...
[정성태] 아래의 글에 정리했으니 참고하세요. C# - Typed D...
[정성태] 간단한 재현 프로젝트라도 있을까요? 저런 식으로 설명만 해...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>단정도/배정도 부동 소수점의 정밀도(Precision)에 따른 형변환 손실</h1> <p> 명백히, 아래의 실수 2개는 다른 값입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 11291153<span style='color: blue; font-weight: bold'>36.790</span> 11291153<span style='color: blue; font-weight: bold'>76.400</span> </pre> <br /> 하지만, 이러한 다름은 8바이트 배정도 실수일 때 그런 것이지, 4바이트 단정도 실수일 때는 값이 같습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // C# 11 + .NET 7 static void Main(string[] args) { float old = 1129115336.790f; Console.WriteLine($"{old:F10}"); float current = 1129115376.400f; Console.WriteLine($"{current:F10}"); } /* 출력 결과 1129115392.0000000000 1129115392.0000000000 */ </pre> <br /> 그리고 이건 IEEE 754 부동 소수점 포맷을 따르는 모든 언어에서 같습니다. 에를 들어 Go 언어에서도 동일한 출력을 확인할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // Go 1.19.4 func main() { var old float32 = 1129115336.790 var current float32 = 1129115376.400 fmt.Printf("%.10f\n", old) fmt.Printf("%.10f\n", current) } /* 출력 결과 1129115392.0000000000 1129115392.0000000000 */ </pre> <br /> 이런 현상이 발생하는 원인은, 배정도 실수의 경우 가수 부분으로 52비트를 할당한 반면, 단정도 실수는 23비트라는 (어쩔 수 없었겠지만) 짧은 정밀도를 가진 탓에 있습니다.<br /> <br /> [단정도 실수 - 그림 출처: <a target='tab' href='https://ko.wikipedia.org/wiki/IEEE_754'>https://ko.wikipedia.org/wiki/IEEE_754</a>]<br /> <img alt='single_float_1.png' src='/SysWebRes/bbs/single_float_1.png' /><br /> <br /> [배정도 실수 - 그림 출처: <a target='tab' href='https://en.wikipedia.org/wiki/Double-precision_floating-point_format'>https://en.wikipedia.org/wiki/Double-precision_floating-point_format</a>]<br /> <img alt='single_float_2.png' src='/SysWebRes/bbs/single_float_2.png' /><br /> <br /> 실제로 우리가 테스트했던 <a target='tab' href='https://www.rapidtables.com/convert/number/decimal-to-binary.html'>10진수 숫자를 2진수로 바꾸면</a> 다음과 같은데요,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // 1129115336.790 0100 0011 0100 1100 1110 1110 1100 1000.1100 1010 0011 1101 0111 // 1129115376.400 0100 0011 0100 1100 1110 1110 1111 0000.0110 0110 0110 0110 0110 </pre> <br /> 앞자리를 1로 놓고 지수를 결정하는 식으로 정규화를 하기 때문에 다음과 같이 23비트에 해당하는 가수가 각각 선택됩니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // 1129115336.790 _100 0011 0100 1100 1110 1110 1... ==> 마지막 자리가 1이므로 반올림 _<span style='color: blue; font-weight: bold'>100 0011 0100 1100 1110 1111</span> // 1129115376.400 _100 0011 0100 1100 1110 1110 1... ==> 마지막 자리가 1이므로 반올림 _<span style='color: blue; font-weight: bold'>100 0011 0100 1100 1110 1111</span> </pre> <br /> 결국, 정수 영역에 해당하는 것조차 23비트 가수 영역으로는 부족한 상태이므로 소수점 영역은 아예 전부 잘려나갔습니다. 이처럼 float32로 표현되면 그 숫자 값이 1129115376이 되어 값이 같아진 것입니다.<br /> <br /> 그리고 원래의 값과 비교했을 때, 10진수의 보존된 값은 앞에서 6자리입니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 위에서 제가 "6자리"의 숫자가 보존되었다고 했는데요, C#의 실수 표현 문서를 보면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > Floating-point numeric types (C# reference) ; <a target='tab' href='https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/builtin-types/floating-point-numeric-types'>https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/builtin-types/floating-point-numeric-types</a> </pre> <br /> <img alt='single_float_3.png' src='/SysWebRes/bbs/single_float_3.png' /><br /> <br /> float(32비트)의 경우 Precision이 "~6-9 digits"라는 문구를 볼 수 있는데, 위에서 테스트한 6자리는 그 기준에 부합합니다. 지수를 생각하지 않는다면 가수부의 23비트는 액면 그대로 10진수 7자리에 해당하기 때문에 아마도 일반적인 경우 6~9자리의 10진수 정도에 해당하는 정밀도가 있다고 하는 것 같습니다. (혹시, 정확하게 이때의 정밀도에 대해 설명해 주실 분이 계실까요? ^^ 덧글 부탁드립니다.) 하지만 가수와 지수를 구분해서 보관한다는 실수 표현의 성격상, 가수에 해당하는 비트만 보관할 수 있다면 경우에 따라 전체 숫자 값이 그대로 보존될 수 있는 여지가 있습니다.<br /> <br /> 가령 위에서 예로 든 "1,129,115,336.790" 값의 정수 부분과 정확히 자릿수가 일치하는 "2,147,483,648" 값은 숫자가 더 큼에도 불구하고 float32로 잘 보존이 됩니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > float t2 = 2147483648f; Console.WriteLine($"{t2:F10}"); // 출력 결과: 2147483648.0000000000 </pre> <br /> 왜냐하면 해당 숫자는 2진수로 이렇고,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 1000 0000 0000 0000 0000 0000 0000 0000 </pre> <br /> 따라서, 단 1비트만 보관할 수 있어도 나머지는 지수로 감당하므로 숫자가 그대로 보존된 것입니다. 따라서 이런 경우에는 10진수 숫자의 모든 값이 잘 보존되었으므로 정밀도는 10이 된 것입니다.<br /> <br /> 물론, 23비트만 만족한다면, 더 큰 숫자도 보존할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > 4,784,511,654,127,730,688 ==> 0100 0010 0110 0110 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000 0000 가수부에 보존할 값 23비트 .100 0010 0110 0110 0000 0000 </pre> <br /> 실제로 직접 C# 코딩으로 테스트해도 결과가 잘 나옵니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > BigInteger b = BigInteger.Parse("4784511654127730688"); float t3 = (float)b; Console.WriteLine($"{t3:F10}"); // 출력 결과: 4784511654127730688.0000000000 </pre> <br /> 저렇게 되면 정밀도는 19가 되는 건가요? ^^<br /> <br /> 반대로, 10진수로는 0.1인 단순한 값조차도 2진수로는 표현할 수 없다는 한계로 인해 부동 소수에서는 제대로 값을 표현하지 못하는 문제도 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // 10진수 0.1 // 2진수 0.0001100110011001100...[1100 반복].... float f1 = 0.1f; Console.WriteLine($"{f1:F70}"); double f2 = 0.1; Console.WriteLine($"{f2:F70}"); /* 출력 결과 0.1000000014901161193847656250000000000000000000000000000000000000000000 0.1000000000000000055511151231257827021181583404541015625000000000000000 */ </pre> <br /> <hr style='width: 50%' /><br /> <br /> 이처럼, 가수부와 지수부에 대한 독립적인 역할로 인해 부동 소수점 데이터 타입의 경우에는 (정수와는 달리) 작은 숫자 범위에서는 형변환 손실이 없을 거라는, 또는 그 반대로 큰 숫자 범위에서는 반드시 형변환 손실이 있을 거라는 가정을 해서는 안 됩니다.<br /> <br /> 암튼, 이런저런 부동 소수점의 특성 때문에 이렇게나 많은 글들을 쓰게 되는군요. ^^<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > C# - 부동 소수 계산 왜 이렇게 나오죠? (1) ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/10872'>https://www.sysnet.pe.kr/2/0/10872</a> C# - 부동 소수 계산 왜 이렇게 나오죠? (2) ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/10873'>http://www.sysnet.pe.kr/2/0/10873</a> double 값을 구할 때는 반드시 피연산자를 double로 형변환! ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/11055'>https://www.sysnet.pe.kr/2/0/11055</a> C#, C++ - double의 Infinity, NaN 표현 방식 ; <a target='tab' href='https://www.sysnet.pe.kr/2/0/11896'>https://www.sysnet.pe.kr/2/0/11896</a> </pre> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
5720
(왼쪽의 숫자를 입력해야 합니다.)