fermat_factorization_method.sf 1.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152
  1. #!/usr/bin/ruby
  2. # Daniel "Trizen" Șuteu
  3. # Date: 22 December 2017
  4. # https://github.com/trizen
  5. # 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 p = n.isqrt
  18. var q = (p*p - n)
  19. while (!q.is_square) {
  20. q += ((p++ << 1) + 1)
  21. }
  22. var s = q.isqrt
  23. var f1 = (p + s)
  24. var f2 = (p - s)
  25. __FUNC__(f1) +
  26. __FUNC__(f2) -> sort
  27. }
  28. for n in ([160587846247027, 5040, 65127835124, 6469693230]) {
  29. var factors = fermat_factorization(n)
  30. say "#{factors.join(' * ')} = #{n}"
  31. assert(factors.all { .is_prime })
  32. assert_eq(factors.prod, n)
  33. }
  34. __END__
  35. 12672269 * 12672383 = 160587846247027
  36. 2 * 2 * 2 * 2 * 3 * 3 * 5 * 7 = 5040
  37. 2 * 2 * 11 * 19 * 6359 * 12251 = 65127835124
  38. 2 * 3 * 5 * 7 * 11 * 13 * 17 * 19 * 23 * 29 = 6469693230