This is an implementation of various combinatorial functions.
These functions always return BigInt
values. This convention
is signaled by the fact that these functions' names begin
with a capital letter.
If we want to calculate 20!, it's easy enough to do this:
julia> factorial(20)
2432902008176640000
However, for 100!, we see this:
julia> factorial(100)
ERROR: OverflowError: 100 is too large to look up in the table
The problem is that 100! is too big to fit in an Int
answer. Of course,
we could resolve this problem this way:
julia> factorial(big(100))
93326215443944152681699238856266700490715968264381621468592963895217599993229915608941463976156518286253697920827223758251185210916864000000000000000000000000
This limitation on factorials causes problems for functions such as stirlings1
in the Combinatorics
package:
julia> using Combinatorics
julia> stirlings1(30,1)
ERROR: OverflowError: 29 is too large to look up in the table; consider using `factorial(big(29))` instead
We take a different approach. We shouldn't have to worry about how large
our arguments may be before a combinatorial function overflows. Instead,
let's assume the result is always of type BigInt
so the calculation
will not be hampered by this problem.
For example:
julia> using BigCombinatorics
julia> Stirling1(30,1)
-8841761993739701954543616000000
When calculating the n
-th Fibonnaci number (or n!
), one implicitly calculates all the
Fibonacci numbers (or factorials) up through n
. This module saves the results of all those calculations so that subsequent invocations of these functions use the previously stored values.
In some cases, the built-in Julia functions with similar names are sufficiently speedy that we don't bother saving the results, but rather simply wrap those functions in ours.
For a single, one-time evaluation of a combinatorial function, the methods in Combinatorics
are likely to be the best option. But for repeated calls to the same function, BigCombinatorics
may perform better:
julia> using Combinatorics, BigCombinatorics
julia> @time x = [bellnum(k) for k=1:1000];
50.067243 seconds (333.34 M allocations: 62.504 GiB, 13.83% gc time)
julia> @time y = [Bell(k) for k=1:1000];
4.222006 seconds (28.25 M allocations: 914.731 MiB, 3.18% gc time, 3.78% compilation time)
julia> @time x = [bellnum(k) for k=1:1000]; # second time is no faster
53.210110 seconds (333.34 M allocations: 62.504 GiB, 14.18% gc time)
julia> @time y = [Bell(k) for k=1:1000]; # values cached so much faster
0.000849 seconds (2.20 k allocations: 42.312 KiB)
julia> x == y
true
Functions such as factorial, Stirling numbers, and so forth obey nice recurrence relations that are mathematically elegant but can be computationally problematic.
When we compute values via these recurrence relations we always save previously computed results and thereby avoid combinatorial explosion. For example:
julia> using Combinatorics, BigCombinatorics
julia> @time stirlings2(34,17)
5.532814 seconds
1482531184316650855 # this is incorrect because arithmetic was done with Int64 values
julia> @time Stirling2(34,17)
0.000920 seconds (3.69 k allocations: 115.836 KiB)
118144018577011378596484455
julia> @time Stirling2(34,17) # second call is even faster because value was cached
0.000011 seconds (2 allocations: 64 bytes)
118144018577011378596484455
For univariate functions, we do not use recursive code and so we avoid stack overflow. (Multivariate functions may still suffer from stack overflows.)
This module is self-contained and does not rely on others. In particular, we use neither Combinatorics
(which provides many of these functions, but with a different design philosopy) nor Memoize
(which also provides caching of previous results but does not give a way to delete stored values).
Fibonacci(n)
returns then
-th Fibonacci number withFibonacci(0)==0
andFibonacci(1)==1
.Factorial(n)
returnsn!
andFactorial(n,k)
returnsn!/k!
.FallingFactorial(n,k)
returnsn*(n-1)*(n-2)*...*(n-k+1)
.RisingFactorial(n,k)
returnsn*(n+1)*(n+2)*...*(n+k-1)
.DoubleFactorial(n)
returnsn!!
.HyperFactorial(n)
returns1^1 * 2^2 * ... * n^n
.Catalan(n)
returns then
-th Catalan number.Derangements(n)
returns the number of derangements of ann
-element set.Binomial(n,k)
returns the number ofk
-element subsets of ann
-element set.MultiChoose(n,k)
returns the number ofk
-element multisets that can be formed using the elements of ann
-element set. Warning: This is not the same asMultinomial
.Multinomial(vals)
returns the multinomial coefficient where the top index is the sum ofvals
. Here,vals
may either be a vector of integers or a comma separated list of arguments. In other words, bothMultinomial([3,3,3])
andMultinomial(3,3,3)
return the multinomial coefficient with top index9
and bottom indices3,3,3
. The result is1680
. Warning: This is not the same asMultiChoose
.Bell(n)
returns then
-th Bell number, i.e., the number of partitions of ann
-element set.Stirling1(n,k)
returns the signed Stirling number of the first kind.Stirling2(n,k)
returns the Stirling number of the second kind, i.e., the number of partitions of ann
-element set intok
-parts (nonempty).Fibonacci(n)
returns then
-th Fibonacci number withFibonacci(0)==0
andFibonacci(1)==1
.IntPartitions(n)
returns the number of partitions of the integern
andIntPartitions(n,k)
returns the number of partitions of the integern
with exactlyk
parts.IntPartitionsDistinct(n)
returns the number of partitions ofn
into distinct parts andIntPartitionsDistinct(n,k)
returns the number of partitions ofn
intok
distinct parts.Euler(n)
returns then
-th Euler number.Eulerian(n,k)
returns the number of permutations of1:n
withk
ascents.PowerSum(n,k)
returns the sum1^k + 2^k + ... + n^k
.Menage(n)
returns the number of solutions to the Menage problem withn
male/female couples.
For most of these functions we save the values we have computed and often values for smaller arguments. For example, when we compute Fibonacci(10)
we have computed and saved the value of Fibonacci(n)
for all values of n
up to 10.
Calling one of these functions with no arguments reinitializes the table of stored values for that function. Most of the stored values are lost.
The function BigCombinatorics.cache_clear()
reinitializes all the tables.
The function BigCombinatorics.cache_report()
prints out the number of values
stored for each function. (Note that some functions don't save any values.)
julia> Bell(10)
115975
julia> Fibonacci(20)
6765
julia> BigCombinatorics.cache_report()
2 Derangements
0 Stirling2
0 Eulerian
1 Euler
3 DoubleFactorial
0 Stirling1
0 PowerSum
2 HyperFactorial
11 Bell
0 IntPartitions
21 Fibonacci
40 Total entries
julia> Fibonacci()
julia> BigCombinatorics.cache_report()
2 Derangements
0 Stirling2
0 Eulerian
1 Euler
3 DoubleFactorial
0 Stirling1
0 PowerSum
2 HyperFactorial
11 Bell
0 IntPartitions
2 Fibonacci
21 Total entries