nth_squarefree.sf 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. #!/usr/bin/ruby
  2. # Daniel "Trizen" Șuteu
  3. # Date: 07 July 2022
  4. # https://github.com/trizen
  5. # Fast algorithm for computing the n-th squarefree number.
  6. # See also:
  7. # https://oeis.org/A005117
  8. # PARI/GP program:
  9. # S(n) = my(s); forsquarefree(k=1,sqrtint(n),s+=n\k[1]^2*moebius(k)); s;
  10. # a(n) = my(min=1, max=231, k=0, sc=0); if(n >= 144, min=floor(zeta(2)*n - 5*sqrt(n)); max=ceil(zeta(2)*n + 5*sqrt(n))); while(min <= max, k=(min+max)\2; sc=S(k); if(abs(sc-n) <= sqrtint(n), break); if(sc > n, max=k-1, if(sc < n, min=k+1, break))); while(!issquarefree(k), k-=1); while(sc != n, my(j=1); if(sc > n, j=-1); k += j; sc += j; while(!issquarefree(k), k+=j)); k;
  11. func nth_squarefree(n) {
  12. n == 0 && return 0 # not squarefree, but...
  13. n <= 0 && return NaN
  14. n == 1 && return 1
  15. var min = 1
  16. var max = 231
  17. # Bounds on squarefree numbers:
  18. # https://mathoverflow.net/questions/66701/bounds-on-squarefree-numbers
  19. if (n >= 268293) {
  20. min = int(zeta(2)*n - 0.058377*sqrt(n))
  21. max = int(zeta(2)*n + 0.058377*sqrt(n))
  22. }
  23. elsif (n >= 144) {
  24. min = int(zeta(2)*n - 5*sqrt(n))
  25. max = int(zeta(2)*n + 5*sqrt(n))
  26. }
  27. var k = 0
  28. var c = 0
  29. loop {
  30. k = (min + max)>>1
  31. c = k.squarefree_count
  32. if (abs(c - n) <= k.isqrt) {
  33. break
  34. }
  35. given (c <=> n) {
  36. when (+1) { max = k-1 }
  37. when (-1) { min = k+1 }
  38. else { break }
  39. }
  40. }
  41. while (!is_squarefree(k)) {
  42. --k
  43. }
  44. while (c != n) {
  45. var j = (n <=> c)
  46. k += j
  47. c += j
  48. k += j while !k.is_squarefree
  49. }
  50. return k
  51. }
  52. for n in (1..10) {
  53. var s = nth_squarefree(10**n)
  54. assert(s.is_squarefree)
  55. assert_eq(s.squarefree_count, 10**n)
  56. assert_eq(10**n -> nth_squarefree, s)
  57. say "S(10^#{n}) = #{s}"
  58. }
  59. assert_eq(
  60. nth_squarefree.map(1..100),
  61. 100.by { .is_squarefree },
  62. )
  63. __END__
  64. S(10^1) = 14
  65. S(10^2) = 163
  66. S(10^3) = 1637
  67. S(10^4) = 16446
  68. S(10^5) = 164498
  69. S(10^6) = 1644918
  70. S(10^7) = 16449369
  71. S(10^8) = 164493390
  72. S(10^9) = 1644934081
  73. S(10^10) = 16449340709
  74. S(10^11) = 164493406178
  75. S(10^12) = 1644934067511
  76. S(10^13) = 16449340668746
  77. S(10^14) = 164493406685659
  78. S(10^15) = 1644934066850410
  79. S(10^16) = 16449340668485215
  80. S(10^17) = 164493406684817902
  81. S(10^18) = 1644934066848209910