ホームページ > バックエンド開発 > PHPチュートリアル > BBP 式を使用して円周率を計算するコード

BBP 式を使用して円周率を計算するコード

WBOY
リリース: 2016-07-25 09:01:23
オリジナル
2707 人が閲覧しました
BBP 式は、円周率の N 番目の結果を直接取得できると主張していますが、取得速度はどの位置でも同じであると考えていましたが、bellard のホームページ (http://bellard.org/pi/) からダウンロードしてください。改良した C プログラムをテストしたところ、N が大きくなるほど計算時間が長くなりました。PHP コードに変換した後、元の C プログラムの計算も 1K 桁目を計算するのに 1 分以上かかりました。遅い。

このプログラムは一度に 9 つの数値のみを出力します (囧rz、より多くの数値を出力するように変更する方法がわかりません)
  1. /**
  2. * 円周率計算 (BBP)
  3. * @author Moyo
  4. * @url http://moyo.uuland.org/code/php-pi-calc/
  5. * @version 1.0
  6. * @日付 2013.01.12
  7. */
  8. class pi
  9. {
  10. public static function calc($__N__)
  11. {
  12. $n = (int)$__N__;
  13. $av = $ a = $vmax = $N = $num = $den = $k = $kq = $kq2 = $t = $v = $s = $i = 0;
  14. $sum = 0.0;
  15. $N = (int) (($n + 20) * log(10) / log(2));
  16. $sum = 0;
  17. for ($a = 3; $a {
  18. $vmax = (int)(log(2 * $N) / log($a));
  19. $av = 1;
  20. for ($i = 0; $i {
  21. $av = ($av * $a);
  22. }
  23. $s = 0;
  24. $num = 1;
  25. $den = 1;
  26. $v = 0;
  27. $ kq = 1;
  28. $kq2 = 1;
  29. for ($k = 1; $k {
  30. $t = $k;
  31. if ($kq >= $a )
  32. {
  33. do
  34. {
  35. $t = (int)($t / $a);
  36. $v --;
  37. }
  38. while (($t % $a) == 0);
  39. $kq = 0 ;
  40. }
  41. $kq ++;
  42. $num = self::mul_mod($num, $t, $av);
  43. $t = (2 * $k -1);
  44. if ($kq2 >= $ a)
  45. {
  46. if ($kq2 == $a)
  47. {
  48. do
  49. {
  50. $t = (int)($t / $a);
  51. $v ++;
  52. }
  53. while (($t % $a) == 0);
  54. }
  55. $kq2 -= $a;
  56. }
  57. $den = self::mul_mod($den, $t, $av);
  58. $kq2 += 2;
  59. if ($ v > 0)
  60. {
  61. $t = self::inv_mod($den, $av);
  62. $t = self::mul_mod($t, $num, $av);
  63. $t = self::mul_mod ($t, $k, $av);
  64. for ($i = $v; $i {
  65. $t = self::mul_mod($t, $a, $ av);
  66. }
  67. $s += $t;
  68. if ($s >= $av)
  69. {
  70. $s -= $av;
  71. }
  72. }
  73. }
  74. $t = self::pow_mod(10 , ($n - 1), $av);
  75. $s = self::mul_mod($s, $t, $av);
  76. $sum = (double)fmod((double)$sum + (double)$ s / (double)$av, 1.0);
  77. }
  78. return array(
  79. 'n' => $n,
  80. 'v' => sprintf('%09d', (int)($sum * 1e9) )
  81. );
  82. }
  83. プライベート静的関数 next_prime($n)
  84. {
  85. do
  86. {
  87. $n ++;
  88. }
  89. while (!self::is_prime($n));
  90. return $n;
  91. }
  92. プライベート静的関数 is_prime($n)
  93. {
  94. $r = $i = 0;
  95. if (($n % 2) == 0)
  96. {
  97. return 0;
  98. }
  99. $r = (int)(sqrt ($n));
  100. for ($i = 3; $i {
  101. if (($n % $i) == 0)
  102. {
  103. return 0;
  104. }
  105. }
  106. return 1;
  107. }
  108. プライベート静的関数 mul_mod($a, $b, $m)
  109. {
  110. return fmod((double)$a * (double)$b, $m);
  111. }
  112. プライベート静的関数 inv_mod($x, $y)
  113. {
  114. $q = $u = $v = $a = $c = $t = 0;
  115. $u = $x;
  116. $v = $y;
  117. $ c = 1;
  118. $a = 0;
  119. do
  120. {
  121. $q = (int)($v / $u);
  122. $t = $c;
  123. $c = $a - $q * $c;
  124. $a = $t;
  125. $t = $u;
  126. $u = $v - $q * $u;
  127. $v = $t;
  128. }
  129. while ($u != 0);
  130. $a = $ a % $y;
  131. if ($a < 0)
  132. {
  133. $a = $y + $a;
  134. }
  135. return $a;
  136. }
  137. プライベート静的関数 pow_mod($a, $b, $m)
  138. {
  139. $r = $aa = 0;
  140. $r = 1;
  141. $aa = $a;
  142. while (1)
  143. {
  144. if ($b & 1)
  145. {
  146. $r = self::mul_mod( $r, $aa, $m);
  147. }
  148. $b = $b >> 1;
  149. if ($b == 0)
  150. {
  151. Break;
  152. }
  153. $aa = self::mul_mod($ aa, $aa, $m);
  154. }
  155. return $r;
  156. }
  157. }
  158. ?>
コードをコピー


ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート