Bernoulli numbers (Erlang)
From LiteratePrograms
The Bernoulli numbers Bm are a sequence of rational numbers defined by the identity
with the initial condition B0 = 1. Here, denotes a binomial coefficient. The sequence, which continues -1/2, 1/6, 0, -1/30, 0, 1/42, ..., obeys no simple pattern, and computing Bm for m as small as a few hundred is a relatively time-consuming task.
Contents |
Rational Numbers
For generating Bernoulli numbers we first need a way to represent rational numbers. We'll make a module for handling regular number functions: add, subtract, multiply, divide, inverse, and negation.
<<rational.erl>>= -module(rational). -export([is_rational/1, make/2, make/1, add/2, sub/2, mult/2, divide/2, inv/1, neg/1]). -record(rational, {top, bot}). gcd(A, 0) -> A; gcd(A, B) -> gcd(B, A rem B). lcm(A, B) -> (A*B) div gcd(A, B). is_rational(X) when record(X, rational) -> true; is_rational(_) -> false. make(Top, Bot) -> Gcd = gcd(Top, Bot), #rational{top = Top div Gcd, bot = Bot div Gcd}. make(Num) when is_integer(Num) -> make(Num, 1); make(Num) when record(Num, rational) -> Num. inv(X) -> A = make(X), make(A#rational.bot, A#rational.top). neg(X) -> A = make(X), make(-A#rational.top, A#rational.bot). mult(X, Y) -> A = make(X), B = make(Y), Top = A#rational.top * B#rational.top, Bot = A#rational.bot * B#rational.bot, Gcd = gcd(Top, Bot), make(Top div Gcd, Bot div Gcd). divide(X, Y) -> mult(X, inv(Y)). add(X, Y) -> A = make(X), B = make(Y), Lcm = lcm(A#rational.bot, B#rational.bot), Top = A#rational.top * (Lcm div A#rational.bot) + B#rational.top * (Lcm div B#rational.bot), make(Top, Lcm). sub(X, Y) -> add(X, neg(Y)).
Memoize
The values of the bernoulli numbers need to be memoized to avoid recalculation over and over of the same number. This should be transparent to the caller of this function. Ets can handle this. We check if the ets table exists and create one if it does not. We then check if the value has already been calculated and return it if it has. Otherwise we calculate it, store it in the ets table, and then return it.
<<memoize>>= bernoulli(N) -> Name = bernoulli, case ets:info(Name) of undefined -> ets:new(Name, [public, named_table]); _ -> true end, case ets:lookup(Name, N) of [] -> Val = bernoulli_i(N), ets:insert(Name, {N, Val}), Val; [{N, Val}] -> Val; Else -> Else end.
Implementation
Now to implement the actual function. The computation can be performed more efficiently according to the following identities due to Srinivasa Ramanujan:
where
<<impl>>= bernoulli_i(0) -> 1; bernoulli_i(1) -> rational:make(-1,2); bernoulli_i(N) when N rem 2 == 1 -> 0; bernoulli_i(N) when N rem 6 == 0 -> Val = rational:sub(rational:make(N+3, 3), bernoulli_a(N, N div 6)), rational:divide(Val, choose(N+3, N)); bernoulli_i(N) when N rem 6 == 2 -> Val = rational:sub(rational:make(N+3, 3), bernoulli_a(N, (N-2) div 6)), rational:divide(Val, choose(N+3, N)); bernoulli_i(N) when N rem 6 == 4 -> Val = rational:sub(rational:make(-(N+3), 6), bernoulli_a(N, (N-4) div 6)), rational:divide(Val, choose(N+3, N)).
Now we implement A:
<<a>>= %This can be sped up by not re-calculating the binary coefficient every time or %memoizing that function. bernoulli_a(N, M) -> bernoulli_a(N, M, 0). bernoulli_a(_N, 0, Sum) -> Sum; bernoulli_a(N, M, Sum) -> Val = rational:mult(choose(N+3, N-6*M), bernoulli(N-6*M)), bernoulli_a(N, M-1, rational:add(Sum, Val)).
The last piece we need is to calculate the binary coefficient:
<<bincoef>>= min(X, Y) when X<Y -> X; min(_X, Y) -> Y. choose(N, R) -> choose(N, min(R, N-R), 1, 1). choose(_N, 0, Top, Bottom) -> Top div Bottom; choose(N, R, Top, Bottom) -> choose(N-1, R-1, Top*N, Bottom*R).
Now we put it all together in a module:
<<bernoulli.erl>>= -module(bernoulli). -export([bernoulli/1]). memoize impl a bincoef
Running
Now the code can be run from the erlang shell:
1> c(rational). {ok,rational} 2> c(bernoulli). {ok,bernoulli} 3> bernoulli:bernoulli(6). {rational,1,42}