count_of_smooth_numbers_mpz.pl 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. #!/usr/bin/perl
  2. # Daniel "Trizen" Șuteu
  3. # Date: 18 July 2020
  4. # https://github.com/trizen
  5. # Count the number of B-smooth numbers <= n.
  6. # See also:
  7. # https://en.wikipedia.org/wiki/Smooth_number
  8. use 5.020;
  9. use ntheory qw(:all);
  10. use experimental qw(signatures);
  11. use Math::GMPz;
  12. sub smooth_count ($n, $k) {
  13. if (ref($n) ne 'Math::GMPz') {
  14. $n = Math::GMPz->new("$n");
  15. }
  16. if ($k < 2 or Math::GMPz::Rmpz_sgn($n) <= 0) {
  17. return 0;
  18. }
  19. if (Math::GMPz::Rmpz_cmp_ui($n, $k) <= 0) {
  20. return $n;
  21. }
  22. my $count = sub {
  23. my ($n, $p) = @_;
  24. if ($p == 2) {
  25. return Math::GMPz::Rmpz_sizeinbase($n, 2);
  26. }
  27. my $t = Math::GMPz::Rmpz_init();
  28. my $q = prev_prime($p);
  29. my $sum = 0;
  30. for (my $k = 0; ; ++$k) {
  31. Math::GMPz::Rmpz_ui_pow_ui($t, $p, $k);
  32. Math::GMPz::Rmpz_tdiv_q($t, $n, $t);
  33. if (Math::GMPz::Rmpz_cmp_ui($t, $q) <= 0) {
  34. $sum += Math::GMPz::Rmpz_get_ui($t);
  35. last;
  36. }
  37. else {
  38. $sum += __SUB__->($t, $q);
  39. }
  40. }
  41. $sum;
  42. }->($n, prev_prime($k + 1));
  43. return $count;
  44. }
  45. foreach my $p (@{primes(50)}) {
  46. say "Ψ(10^n, $p) for n <= 10: [", join(', ', map { smooth_count(powint(10, $_), $p) } 0 .. 10), "]";
  47. }
  48. __END__
  49. Ψ(10^n, 2) for n <= 10: [1, 4, 7, 10, 14, 17, 20, 24, 27, 30, 34]
  50. Ψ(10^n, 3) for n <= 10: [1, 7, 20, 40, 67, 101, 142, 190, 244, 306, 376]
  51. Ψ(10^n, 5) for n <= 10: [1, 9, 34, 86, 175, 313, 507, 768, 1105, 1530, 2053]
  52. Ψ(10^n, 7) for n <= 10: [1, 10, 46, 141, 338, 694, 1273, 2155, 3427, 5194, 7575]
  53. Ψ(10^n, 11) for n <= 10: [1, 10, 55, 192, 522, 1197, 2432, 4520, 7838, 12867, 20193]
  54. Ψ(10^n, 13) for n <= 10: [1, 10, 62, 242, 733, 1848, 4106, 8289, 15519, 27365, 45914]
  55. Ψ(10^n, 17) for n <= 10: [1, 10, 67, 287, 945, 2579, 6179, 13389, 26809, 50351, 89679]
  56. Ψ(10^n, 19) for n <= 10: [1, 10, 72, 331, 1169, 3419, 8751, 20198, 42950, 85411, 160626]
  57. Ψ(10^n, 23) for n <= 10: [1, 10, 76, 369, 1385, 4298, 11654, 28434, 63768, 133440, 263529]
  58. Ψ(10^n, 29) for n <= 10: [1, 10, 79, 402, 1581, 5158, 14697, 37627, 88415, 193571, 399341]
  59. Ψ(10^n, 31) for n <= 10: [1, 10, 82, 434, 1778, 6070, 18083, 48366, 118599, 270648, 581272]
  60. Ψ(10^n, 37) for n <= 10: [1, 10, 84, 461, 1958, 6952, 21535, 59867, 152482, 361173, 804369]
  61. Ψ(10^n, 41) for n <= 10: [1, 10, 86, 485, 2129, 7833, 25133, 72345, 190767, 467495, 1076462]
  62. Ψ(10^n, 43) for n <= 10: [1, 10, 88, 508, 2300, 8740, 28955, 86086, 234423, 592949, 1408465]
  63. Ψ(10^n, 47) for n <= 10: [1, 10, 90, 529, 2463, 9639, 32876, 100688, 282397, 735425, 1797897]