ホームページ >バックエンド開発 >PHPチュートリアル >php_PHP チュートリアルの浮動小数点計算の問題

php_PHP チュートリアルの浮動小数点計算の問題

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBオリジナル
2016-07-13 10:39:27756ブラウズ

PHP の +-*/ を使用して浮動小数点数を計算すると、不正確な計算結果による問題が発生する可能性があります。たとえば、echo intval(0.58*100); は 58 ではなく 57 を出力します。これは実際には基礎となるバイナリです。浮動小数点数を正確に表現できないバグは、Python を使用してもこの問題に遭遇しました。したがって、基本的にほとんどの言語は、正確な計算を行うためのクラス ライブラリまたは関数ライブラリを提供しています。たとえば、PHP には BC 高精度関数ライブラリがあります。以下では、PHP トレーニング教師のデーンが、よく使用されるいくつかの BC 高精度関数の使用方法を紹介します。 。

コードは次のとおりです

出力が 57 なのはなぜですか?

bugs.php.net でよく質問されることは言うまでもなく、同様の質問をする人がたくさんいるので、多くの学生がこの質問を抱いたことがあると思います...

この理由を理解するには、まず浮動小数点数の表現 (IEEE 754) を知る必要があります:

浮動小数点数は、64 ビット長 (倍精度) を例として、1 つの符号ビット (E)、11 の指数ビット (Q)、および 52 ビットの仮数ビット (M) (合計 64 ビット) で表されます。 )。

符号ビット:最上位ビットはデータの符号を表し、0は正の数、1は負の数を表します。

指数ビット:データを基数2で累乗し、指数をオフセットコードで表すことを示します

仮数部:データの小数点以下の有効数字を示します。

ここで重要なのは、小数を二進数で表現することです。小数点が二進数でどのように表現されるかについては、ここでは詳しく説明しません。理解する必要があるのは、0.58 が無限に長いということです。バイナリ表現の場合 (以下の数字は暗黙の 1 を省略します)。

0.58 (52 ビット) のバイナリ表現は基本的に: 00101000111101011100001010001111010111000010100011110.57 バイナリ表現 (52 ビット) は基本的に: 0010001111010111000010100 011 11010111000010100011110 そして、これら 52 ビットだけで計算すると、この 2 つの 2 進数は次のようになります: www.111cn.net

0.58 -> 0.579999999999999960.57 -> 0.5699999999999999 0.58 * 100 の具体的な浮動小数点の乗算については、詳しくは考えませんので、興味のある方は見てください(浮動小数点)。暗算でなんとなく… 0.58 * 100 = 57.999999999

そうすると、計算すると自然に57になります…

この問題の重要なポイントは、「有限の小数を持っているように見えますが、コンピューターの 2 進表現では無限です」であることがわかります。

だから、これはもう PHP のバグだと思わないでください、これがそういうものなのです…

PHP 浮動小数点型 +-*%/

には不正確さの問題があります

例:

1.

$a = 0.1;

$b = 0.7;

var_dump(($a + $b) == 0.8);

出力される値はブール値 false です

これはなぜですか? PHP マニュアルには、浮動小数点数に関する次の警告メッセージがあります。

警告

浮動小数点精度

0.1 や 0.7 などの一見単純な小数は、精度を少し損なうことなく内部バイナリ形式に変換できません。これにより、結果が混乱する可能性があります。たとえば、floor((0.1+0.7)*10) は、結果の内部表現が 7.9999999999… のようなものになるため、通常、期待される 8 ではなく 7 を返します。

これは、有限の桁数で特定の小数を正確に表現することが不可能であるという事実に関係しています。たとえば、10 進数の 1/3 は 0.3333333 になります。

したがって、浮動小数点数の結果が最後の桁まで正確であるとは決して信じないでください。また、2 つの浮動小数点数が等しいかどうかを比較しないでください。本当により高い精度が必要な場合は、任意精度の数学関数または gmp 関数を使用する必要があります

コードは次のとおりです

$a = 0.1; $b = 0.7;
var_dump(bcadd($a,$b,2) == 0.8);



bcadd — 2 つの高精度数値を加算します

bccomp — 2 つの高精度数値を比較し、-1、0、1 を返します

bcdiv — 2 つの高精度数値の除算

bcmod — 高精度の数値の余りを求める

bcmul — 2 つの高精度数値の乗算

bcpow — 高精度の数値の累乗を求めます

bcpowmod — 数論で非常に一般的に使用される、高精度の数値べき乗と係数を求めます

bcscale — デフォルトの小数点数を設定します。これは、Linux bc の "scale="

に相当します。

bcsqrt — 高精度の数値の平方根を求めます

bcsub — 2 つの高精度数値を減算します

いくつかの例をまとめました

php BC 高精度関数ライブラリには、加算、比較、除算、減算、剰余、乗算、n 乗、デフォルトの小数点数の設定、および 2 乗が含まれます。これらの関数は、電子商取引の価格計算などの金銭計算の場合にさらに便利です。

注:設定した桁数については、四捨五入ではなく超過分が切り捨てられます。

www.bkjia.comtru​​ehttp://www.bkjia.com/PHPjc/730239.html技術記事 PHP の +-*/ を使用して浮動小数点数を計算すると、たとえば、echo intval(0.58*100); では 58 ではなく 57 が出力されるなど、問題が発生する可能性があります。これは実際にはコンピュータの結果です。 ...
コードは次のとおりです
/**
* 2 つの高精度数値を比較します
*
* @アクセスグローバル
* @param float $left
* @param float $right
* @param int $scale 正確には小数点以下の桁数
*
* @return int $left==$right は 0 を返します | $left>$right は 1 を返します
​*/
var_dump(bccomp($left=4.45, $right=5.54, 2));
// -1

/**
* 2つの高精度数値を追加します
*
* @アクセスグローバル
* @param float $left
* @param float $right
* @param int $scale 正確には小数点以下の桁数
*
* @return 文字列
​*/
var_dump(badd($left=1.0321456, $right=0.0243456, 2));
//1.04

/**
* 2 つの高精度数値を減算します
*
* @アクセスグローバル
* @param float $left
* @param float $right
* @param int $scale 正確には小数点以下の桁数
*
* @return 文字列
​*/
var_dump(bcsub($left=1.0321456, $right=3.0123456, 2));
//-1.98

/**
* 2 つの高精度数値を除算します
*
* @アクセスグローバル
* @param float $left
* @param float $right
* @param int $scale 正確には小数点以下の桁数
*
* @return 文字列
​*/
var_dump(bcdiv($left=6, $right=5, 2));
//1.20

/**
* 2 つの高精度数値を乗算します
*
* @アクセスグローバル
* @param float $left
* @param float $right
* @param int $scale 正確には小数点以下の桁数
*
* @return 文字列
​*/
var_dump(bcmul($left=3.1415926, $right=2.4569874566, 2));
//7.71

/**
* bc関数の小数点以下の桁数を設定します
*
* @アクセスグローバル
* @param int $scale 正確には小数点以下の桁数
*
* @return void
​*/
bcscale(3);
var_dump(bcdiv('105', '6.55957')); // 16.007
声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。