1. PHP浮動小数点数の精度損失問題
まず次のコードを見てください:
コードをコピーします コードは次のとおりです:
$f = 0.57;
エコー間隔($f * 100); //56
結果は少し驚くかもしれませんが、PHP は IEEE 754 倍精度に従います。
浮動小数点数は、1 つの符号ビット (E)、11 の指数ビット (Q)、および 52 ビットの仮数ビット (M) (合計 64 ビット) を使用して、64 ビットの倍精度で表されます。
符号ビット: 最上位ビットはデータの符号を表し、0 は正の数を表し、1 は負の数を表します。
指数ビット: データが基数 2 乗されていることを示し、指数はオフセット コードで表されます
仮数部: データの小数点以下の有効数字を示します。
10 進数が 2 進数でどのように表現されるかを見てみましょう:
2を掛けて切り上げ、順番に並べます。つまり、小数部を2倍し、整数部を取り、残りの小数部を2倍し、整数部を取り、残りの小数部を2を計算して小数部の部分を取りますが、このように0.57のような小数を掛けると、小数部が0になることはありません。有効数字の10進表現は2進数では無限大です。
0.57 のバイナリ表現は基本的に (52 ビット): 0010001111010111000010100011110101110000101000111101
52 ビットしかない場合、0.57 => 0.56999999999999995
上記の予期せぬ結果を確認するのは難しくありません。
2. PHP浮動小数点数の精度の問題
まず質問を見てみましょう:
コードをコピーします コードは次のとおりです:
$f = 0.58;
var_dump(intval($f * 100)); //なぜ 57 が出力されるのですか
このような疑問を持ったことのある学生さんも多いと思います。
具体的な原理については、「Brother Bird」の記事で詳しい説明があります: PHP 浮動小数点数に関するよくある質問への回答
それでは、この問題を回避するにはどうすればよいでしょうか?
方法はたくさんありますが、ここでは 2 つ紹介します:
1.スプリント
コードをコピーします コードは次のとおりです:
substr(sprintf("%.10f", ($a/ $b)), 0, -7);
2.round(丸くなるので注意)
コードをコピーします コードは次のとおりです:
ラウンド($a/$b, 3);
または、より良い方法がある場合は、メッセージを残して教えてください。
3. PHP 浮動小数点数に関するよくある質問への回答
PHP の浮動小数点数については、以前に次の記事を書きました: PHP の float についてのすべての「偽り」 (PHP の float についてのすべての「偽り」)
しかし、私は次のよくある質問に対する答えを 1 つ見逃していました:
コードをコピーします コードは次のとおりです:
$f = 0.58;
var_dump(intval($f * 100)); //なぜ 57 が出力されるのですか
?>
出力が 57 になるのはなぜですか? PHP のバグですか?
bugs.php.net でよく質問されることは言うまでもなく、同様の質問をする人がたくさんいるので、多くの学生がそのような質問をしたことがあると思います...
この理由を理解するには、まず浮動小数点数の表現 (IEEE 754) を知る必要があります。
浮動小数点数は、64 ビット長 (倍精度) を例として、1 つの符号ビット (E)、11 の指数ビット (Q)、および 52 ビットの仮数ビット (M) で表されます (合計 64 ビット)。 .
符号ビット: 最上位ビットはデータの符号を表し、0 は正の数を表し、1 は負の数を表します。
指数ビット: データの累乗を基数 2 で表し、指数はオフセット コードで表されます
仮数: データの小数点以下の有効数字を示します。
ここで重要なのは、バイナリでの小数を表現する方法については、ここでは詳しく説明しません。0.58 は無限に長いということです。バイナリ表現 (以下の数字は暗黙の 1 を省略します)..
0.58 のバイナリ表現は基本的に (52 ビット): 0010100011110101110000101000111101011100001010001111
0.57 のバイナリ表現は基本的に (52 ビット): 0010001111010111000010100011110101110000101000111101
これら 52 ビットのみを使用して 2 つの 2 進数を計算すると、次のようになります。
コードをコピーします コードは次のとおりです:
0.58 -> 0.57999999999999996
0.57 -> 0.56999999999999995
具体的な浮動小数点の乗算0.58 * 100については、詳しくは考えませんので、興味のある方は暗算で見てみてください... 0.58 * 100 = 57.999999999
それでは、計算してみると、当然57になります…
この問題の重要なポイントは、「一見有限に見える 10 進数は、コンピューターの 2 進数表現では実際には無限である」ということです。
だから、これはもう PHP のバグだと思わないでください。これが実際のバグです…