fermat_factorization_method_2.sf 966 B

123456789101112131415161718192021222324252627282930313233343536373839404142434445
  1. #!/usr/bin/ruby
  2. # Daniel "Trizen" Șuteu
  3. # Date: 16 March 2018
  4. # https://github.com/trizen
  5. # A simple implementation of Fermat's factorization method.
  6. # See also:
  7. # https://en.wikipedia.org/wiki/Fermat%27s_factorization_method
  8. func fermat_factorization (n) {
  9. # Check for primes and negative numbers
  10. return [] if (n <= 1)
  11. return [n] if n.is_prime
  12. # Check for divisibility by 2
  13. if (n.is_even) {
  14. var v = n.valuation(2)
  15. return (v.of(2) + __FUNC__(n >> v))
  16. }
  17. var q = 2*n.isqrt
  18. while (!is_square(q*q - 4*n)) {
  19. q += 2
  20. }
  21. # For a semiprime `n`, we have:
  22. # say "euler_phi(#{n}) = #{n - q + 1}"
  23. var p = (q + isqrt(q*q - 4*n))>>1
  24. __FUNC__(n/p) +
  25. __FUNC__(p) -> sort
  26. }
  27. for n in ([160587846247027, 5040, 65127835124, 6469693230]) {
  28. var factors = fermat_factorization(n)
  29. say "#{factors.join(' * ')} = #{n}"
  30. assert(factors.all { .is_prime })
  31. assert_eq(factors.prod, n)
  32. }