2進数の小数は、
これでは使えないので、内部で有能なダンゴムシが近似値に丸める。ダンゴムシは丸まるだけだ。丸めるのはフンコロガシだった。
ともかく、この結果として生まれるのが丸め誤差である。
0.1+0.2は0.3にならない
丸め誤差のサンプルとして、0.1 + 0.2
という式がよく挙げられる。ハッピー・ウェディング。この式をプログラムで実行すると、結果が0.3
にならない。そんな話がまことしやかに囁かれている。
例えば以下のコードを試してみる。
ふつうに0.300000
が出てくる。
「この嘘つきどもめが!」と怒ってはいけない。これはC言語の%f
がデフォルトでは小数点以下6桁までしか表示しないためである。桁数を増やすと結果が変わる。
17桁目に4
が現れる。
これは、浮動小数点数の仕様を標準化する規格に準ずる多くの言語が共有する挙動である。
2進数では10進数の小数を正確に表現できない
整数でうまくやれているのだから、小数を整数として解釈し直せば、誤差なく計算できそうである。
例えば0.1
を2進数で表現しようとすると無限循環小数となり、丸め誤差が生まれる。最初に10
を掛けて1
にし、必要な演算をした後に10
で割れば、誤差は生まれない。
などと考えると迷宮に迷い込み、時間を無駄にする。いつから0.1
が10進数の正確な0.1
だと錯覚していた?
丸め誤差が生まれるタイミング
0.1 + 0.2
の例が頭にあると混乱するかと思う。しかし丸め誤差は、演算の結果生まれるものではない。コンピュータ上で小数を表現した瞬間に生まれる。なぜならコンピュータの資源は有限であり、無限の小数を表現できないからだ。どこかで丸めて扱うしかない。さもないと亀も追い抜けない。
この事実は、表示する小数の桁数を増やせば、どの言語でもかんたんに確かめられる。
もちろん、0.5
や0.25
のように、2進数できっちり表現できる小数もある。ただ、小数全体からすれば微々たるものである。
コンピュータ上で小数を表現するのはムズい。丸め誤差の問題に加え、桁数が一定ではないという問題もある。こうした難題の解決策として考案されたのが、固定小数点数や浮動小数点数である。
固定小数点数
小数は、最小の桁が一定ではない。例えば32.5
と3.25
では最小の桁が異なる。正しく演算するには、どの桁がどの桁に対応しているかを知る必要がある。こうした処理は、初期のコンピュータには重すぎた。
そのため、計算の前に小数点の位置を定義する方法が取られた。この方法で表現される小数は、固定小数点数と呼ばれる。固定小数点数は、全体を1つの整数として処理し、小数点の位置を戻す。
つまり、先に触れたような処理を、もっとスマートに実現する。ただ、これだと計算できる桁数が限られる。整数部分を増やすと小数部分が減るし、小数部分を増やすと整数部分が減る。演算ごとに小数点の位置を定義する手間もある。
浮動小数点数
浮動小数点数は、固定小数点数よりも精度の高い数値の計算を行うために発明された。
現在では、IEEE 754-2008によってその仕様が標準化されている。IEEE 754の浮動小数点数では、2進数を符号部、指数部、仮数部と分けて表現する。
各部の名称 | 概要 |
---|---|
符号ビット | 最初のビット。0 で正、1 で負を表す。 |
指数部 | 仮数部を元に戻すために必要なビットシフト数。 ……と見せかけて、シフト数に対して仮数部の最初のビット以外を 1 埋めした値を足した値が入る。正負を表現することが主な目的。 |
仮数部 | 対象の値を2進数に直し、1. で始まるようにビットシフトした値。……のお尻を所定のビット数分 0 で埋めた値。 |
これらに割り振られるビット数は、小数の精度によって異なる。IEEE 754標準で規格化され、広く普及しているのは以下の2つのパターンである。
格納形式 | 符号部 | 指数部 | 仮数部 | 全体 |
---|---|---|---|---|
単精度 | 1 | 8 | 23 | 32 |
倍精度 | 1 | 11 | 52 | 64 |
例えば0.0625
という小数を、単精度(32ビット)の浮動小数点数で考える。0.0625
は、2進数だと0.0001
である。
まずこれを1.0
の形に直す。-4
なので、2進数で1111 1100
としたい。
ただ、指数部に正と負の値があると演算が大変だ。全部の正の値であれば演算が楽になる。このために、バイアス値と呼ばれる値が足される。つまり元の正負の値の範囲を丸ごと0以上にズラす。これは、最初の1桁以降を0
で埋めたビット列である。
32ビットの浮動小数点数は指数部が8
ビットであるので、0111 1111
が足される。
また、仮数部は23ビットであるから、1
に22個の0を繋げた値を格納したくなる。しかし、有効桁数を増やすために最初の1
は省略される。1
であることが前提であるから保存する必要はない、ということだ。この隠される1ビットは、隠れビット(hidden bit)と呼ばれる。
このような仕組みのため、今回のケースでは仮数部に23個の0
が並ぶ。
わけわかんないよね。
浮動小数点数と丸め誤差
丸め誤差は、主に浮動小数点数で生まれる。精度が期待される状況では固定小数点数を使うのが一般的だ。
ただ、固定小数点数では丸め誤差が生じないわけではない。全体を整数として解釈しても、2進数を10進数に直すと小数部分では丸め誤差が生じる。
精度が期待される状況で固定小数点数を使うのは、丸め誤差の範囲を予測して切り捨てるためである。数値の細かさという意味の精度であれば、浮動小数点数の方がずっと高い。
これはなんの記事だったか。
0.1 + 0.2
が0.3
にならないのは、2進数と10進数で、表現できる小数が完全に対応していないためである。小数って不思議。