-
Notifications
You must be signed in to change notification settings - Fork 34
/
sum_of_digits_subquadratic_algorithm_mpz.pl
executable file
·60 lines (43 loc) · 1.44 KB
/
sum_of_digits_subquadratic_algorithm_mpz.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
#!/usr/bin/perl
# Subquadratic algorithm for computing the sum of digits of a given integer in a given base.
# Based on the FastIntegerOutput algorithm presented in the book:
#
# Modern Computer Arithmetic
# - by Richard P. Brent and Paul Zimmermann
#
use 5.020;
use strict;
use warnings;
use Math::GMPz;
use ntheory qw(:all);
use experimental qw(signatures);
sub FastSumOfDigits ($A, $B) {
$A = Math::GMPz->new("$A");
# Find k such that B^(2k - 2) <= A < B^(2k)
my $k = (logint($A, $B) >> 1) + 1;
my $Q = Math::GMPz::Rmpz_init();
my $R = Math::GMPz::Rmpz_init();
sub ($A, $k) {
if (Math::GMPz::Rmpz_cmp_ui($A, $B) < 0) {
return Math::GMPz::Rmpz_get_ui($A);
}
my $w = ($k + 1) >> 1;
my $t = Math::GMPz::Rmpz_init();
Math::GMPz::Rmpz_ui_pow_ui($t, $B, $k);
Math::GMPz::Rmpz_divmod($Q, $R, $A, $t);
Math::GMPz::Rmpz_set($t, $Q);
__SUB__->($R, $w) + __SUB__->($t, $w);
}->($A, $k);
}
foreach my $B (2 .. 300) { # run some tests
my $N = factorial($B); # int(rand(~0));
my $x = vecsum(todigits($N, $B));
my $y = FastSumOfDigits($N, $B);
if ($x != $y) {
die "Error for FastSumOfDigits($N, $B): $x != $y";
}
}
say join ', ', FastSumOfDigits(5040, 10); #=> 9
say join ', ', FastSumOfDigits(5040, 11); #=> 20
say join ', ', FastSumOfDigits(5040, 12); #=> 13
say join ', ', FastSumOfDigits(5040, 13); #=> 24