684 Inverse Digit Sum.pl 1.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566
  1. #!/usr/bin/perl
  2. # Daniel "Trizen" Șuteu
  3. # Date: 27 November 2019
  4. # https://github.com/trizen
  5. # See OEIS sequence:
  6. # https://oeis.org/A051885
  7. # The smallest numbers whose sum of digits is n, are numbers of the form r*10^j-1, with r=1..9 and j >= 0.
  8. # This solution uses the following formula:
  9. # Sum_{j=0..n} (r*10^j-1) = (r * 10^(n+1) - r - 9*n - 9)/9
  10. # By letting r=1..9, we get:
  11. # R(k) = Sum_{r=1..9} Sum_{j=0..n} (r*10^j-1) = 2*(2^n * 5^(n+2) - 7) - 9*n
  12. # From R(k), we get S(k) as:
  13. # S(k) = R(k) - Sum_{j=2+(k mod 9) .. 9} (j*10^n-1)
  14. # S(k) = R(k) - (10-r) * (10^n * (r+9) - 2)/2
  15. # Simplifying the formula, we get:
  16. # S(k) = (((r-1)*r + 10) * 10^n - 2*(r + 9*n + 4))/2
  17. # where:
  18. # n = floor(k/9)
  19. # r = 2+(k mod 9)
  20. # https://projecteuler.net/problem=684
  21. # Runtime: 0.031s
  22. use 5.020;
  23. use warnings;
  24. use ntheory qw(:all);
  25. use experimental qw(signatures);
  26. my $MOD = 1000000007;
  27. sub fib ($n) {
  28. lucasu(1, -1, $n);
  29. }
  30. sub S($k) {
  31. my $n = divint($k, 9);
  32. my $r = divrem($k, 9) + 2;
  33. my $x = mulmod(mulmod($r-1, $r, $MOD) + 10, powmod(10, $n, $MOD), $MOD);
  34. my $y = mulmod(2, 4 + $r + mulmod(9, $n, $MOD), $MOD);
  35. my $z = invmod(2, $MOD);
  36. mulmod($x-$y, $z, $MOD);
  37. }
  38. S(20) == 1074 or die "error";
  39. S(49) == 1999945 or die "error";
  40. my $sum = 0;
  41. foreach my $k (2 .. 90) {
  42. $sum = addmod($sum, S(fib($k)), $MOD);
  43. }
  44. say $sum;