-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpartial_sums_of_dedekind_psi_function_recursive.sf
117 lines (83 loc) · 2.62 KB
/
partial_sums_of_dedekind_psi_function_recursive.sf
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
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
#!/usr/bin/ruby
# Daniel "Trizen" Șuteu
# Date: 08 March 2019
# https://github.com/trizen
# Recursive formula for computing partial sums of the Dedekind psi function.
# See also:
# https://en.wikipedia.org/wiki/Dedekind_psi_function
# https://trizenx.blogspot.com/2018/11/partial-sums-of-arithmetical-functions.html
func dedekind_psi_partial_sum (n, m) {
var lookup_size = (2 + 2*n.iroot(3)**2)
var omega_sum_lookup = [0]
for k in (1..lookup_size) {
omega_sum_lookup[k] = (omega_sum_lookup[k-1] + 2**(k.omega))
}
var mu = moebius(0, n.isqrt)
func R (n) is cached { # A064608(n) = Sum_{k=1..n} 2^omega(k)
if (n <= lookup_size) {
return omega_sum_lookup[n]
}
var total = 0
for k in (1..n.isqrt) {
total += mu[k]*(
2*sum(1..isqrt(n / k**2), {|j|
floor(n / (j * k**2))
}) - isqrt(n / k**2)**2
) if mu[k]
}
return total
}
func D (n) is cached { # Sum_{k=1..n} Sum_{d|k} 2^omega(k) * (k/d)^m
var s = n.isqrt
var total = 0
for k in (1..s) {
total += (2**omega(k) * faulhaber(floor(n/k), m))
total += (k**m * R(floor(n/k)))
}
total -= R(s)*faulhaber_sum(s, m)
return total
}
var dedekind_sum_lookup = [0]
for k in (1..lookup_size) {
dedekind_sum_lookup[k] = (dedekind_sum_lookup[k-1] + k.dedekind_psi(m))
}
var cache = Hash()
func (n) {
if (n <= lookup_size) {
return dedekind_sum_lookup[n]
}
if (cache.has(n)) {
return cache{n}
}
var A = D(n)
var B = sum(2 .. floor(n/(1+n.isqrt)), {|k|
__FUNC__(floor(n/k))
})
var C = sum(1 .. n.isqrt, {|k|
(floor(n/k) - floor(n/(k+1))) * __FUNC__(k)
})
cache{n} = (A - B - C)
}(n)
}
func dedekind_psi_partial_sum_test (n, m) { # just for testing
1..n -> sum {|k| dedekind_psi(k, m) }
}
for m in (0 .. 10) {
var n = 100.irand
var t1 = dedekind_psi_partial_sum(n, m)
var t2 = dedekind_psi_partial_sum_test(n, m)
assert_eq(t1, t2)
say "Sum_{k=1..#{n}} ψ_#{m}(k) = #{t1}"
}
__END__
Sum_{k=1..96} ψ_0(k) = 345
Sum_{k=1..47} ψ_1(k) = 1686
Sum_{k=1..29} ψ_2(k) = 9998
Sum_{k=1..43} ψ_3(k) = 962082
Sum_{k=1..31} ψ_4(k) = 6404948
Sum_{k=1..92} ψ_5(k) = 106212512590
Sum_{k=1..41} ψ_6(k) = 30483567832
Sum_{k=1..41} ψ_7(k) = 1102280090206
Sum_{k=1..27} ψ_8(k) = 997166331596
Sum_{k=1..55} ψ_9(k) = 27720063080249258
Sum_{k=1..81} ψ_10(k) = 95772724544494382222