This package offers vector, set and dictionary types that can hold
any number of elements up to some (small) limit. Most of these types come
in a mutable and an immutable version. With bit type elements, the immutable
versions don't allocate. The mutable ones usually require bit type elements
for full functionality.
This approach leads to significant speed-ups compared to the standard types.
For example, the performance of the vector types SmallVector
and MutableSmallVector
is close to that to SVector
and MVector
from
StaticArrays.jl.
The most extreme performance gain is with SmallBitSet
, which can be 100x faster than BitSet
.
(The exact speed-ups depend much on your processor.)
Note: The former submodule Combinatorics
has been turned into the separate package
SmallCombinatorics.jl.
The full documentation for SmallCollections
is available
here.
The types
SmallVector{N,T}
and
MutableSmallVector{N,T}
can hold up to N
elements of type T
.
In many cases they are almost as fast as fixed-length vectors.
Internally, these types use fixed-length vectors and fill the unused elements with some
default
value. The underlying types
FixedVector{N,T}
and
MutableFixedVector{N,T}
store exactly N
elements of type T
.
Up to a few implementation details, they are analogous to SVector
and MVector
.
(However, broadcasting with assignment is currently much faster for MutableSmallVector
than for MVector
.)
Benchmarks for (Mutable)SmallVector
and (Mutable)FixedVector
The timings are for 1000 operations of the given type on vectors having between 1 and 31 elements (or exactly 32 elements for fixed-length vectors). If possible, mutating operations were used.
N = 32 , T = Int16 |
v + w |
v .+= w |
sum |
push(!) |
count(>(c), v) |
---|---|---|---|---|---|
Vector{T} |
45.893 μs | 20.441 μs | 8.403 μs | 5.104 μs | 9.860 μs |
MVector{N,T} |
5.770 μs | 19.196 μs | 2.370 μs | 9.866 μs | 2.801 μs |
MutableFixedVector{N,T} |
2.932 μs | 4.494 μs | 3.086 μs | n/a | 1.492 μs |
MutableSmallVector{N,T} |
3.276 μs | 4.975 μs | 2.977 μs | 3.154 μs | 1.871 μs |
SVector{N,T} |
1.979 μs | n/a | 1.396 μs | 1.988 μs | 1.218 μs |
FixedVector{N,T} |
2.054 μs | n/a | 2.661 μs | n/a | 1.217 μs |
SmallVector{N,T} |
2.517 μs | n/a | 2.661 μs | 6.408 μs | 1.617 μs |
Notes: sum
for SVector
and MVector
returns an Int16
instead of Int
. SmallCollections
has a separate function sum_fast
for this. Addition allocates for Vector
. To avoid this for
MVector
, the result was transformed to SVector
. push!
for MVector
and push
for SVector
are not directly comparable to the others because they change the type of the returned vector,
which leads to type instabilities in cases like loops.
For the benchmark code see the file benchmark/benchmark_vec.jl
in the repository.
How broadcasting or functions like
map
or
any
deal with default values for a (Mutable)SmallVector
is governed by a trait called
MapStyle
.
Except for broadcasting one can override the automatically chosen style.
A
PackedVector{U,M,T}
uses the unsigned integer type U
to store integers with M
bits,
which to the outside appear as having integer type T
. For example, a PackedVector{UInt128,5,Int8}
can store 25
(128÷5
) signed integers with 5
bits. When retrieving them, they are of type Int8
.
Indexing and algebraic operations are usually slower for PackedVector
than for SmallVector
,
while push
/pushfirst
and pop
/popfirst
may be faster.
The types
SmallSet{N,T}
and
MutableSmallSet{N,T}
can hold up to N
elements of type T
. The type
SmallBitSet{U}
can hold integers between 1
and M
where M
is the bit size of the unsigned integer type U
(taken to be UInt
if omitted).
For small integers, MutableSmallSet
is as fast (or even faster than) BitSet
.
SmallBitSet
can be up to 100x faster than BitSet
.
Benchmarks for SmallSet
, MutableSmallSet
and SmallBitSet
The timings are for 1000 operations of the given type on sets having between 1 and 8 elements.
If possible, mutating operations were used.
Note that while a BitSet
can hold arbitrarily many elements, the timings for MutableSmallSet
wouldn't change if the elements were drawn from Int16
without restrictions.
N = 16 , T = Int16 |
push(!) |
intersect(!) |
issubset |
in |
---|---|---|---|---|
Set{T} |
15.669 μs | 88.370 μs | 23.482 μs | 1.610 μs |
MutableSmallSet{N,T} |
8.269 μs | 14.422 μs | 4.357 μs | 1.658 μs |
SmallSet{N,T} |
12.688 μs | 15.011 μs | 10.234 μs | 2.122 μs |
BitSet |
17.481 μs | 23.403 μs | 7.407 μs | 1.836 μs |
SmallBitSet{UInt16} |
1.377 μs | 31.240 ns | 56.226 ns | 1.067 μs |
For the benchmark code see the file benchmark/benchmark_set.jl
in the repository.
The types
SmallDict{N,K,V}
and
MutableSmallDict{N,K,V}
can hold up to N
entries with key type K
and value type V
.
Operations for MutableSmallDict
are typically faster than for Dict
.
For SmallDict
this holds often, but not always.
Since keys and values are stored as small vectors, inverse lookup with
invget
should be as fast as regular lookup.
Benchmarks for SmallDict
and MutableSmallDict
The timings are for 1000 operations of the given type on dictionaries created with 30 randomly chosen key-value pairs. If possible, mutating operations were used.
N = 32 , K = V = Int8 |
getindex |
invget |
setindex(!) |
pop(!) |
iterate |
---|---|---|---|---|---|
Dict{K,V} |
10.799 μs | n/a | 26.992 μs | 18.931 μs | 404.342 μs |
MutableSmallDict{N,K,V} |
4.273 μs | 5.325 μs | 8.900 μs | 7.213 μs | 17.104 μs |
SmallDict{N,K,V} |
1.856 μs | 8.775 μs | 9.823 | 20.368 μs | 15.840 μs |
Note: For SmallDict
, invget
appears much slower than getindex
in this benchmark.
(This started with commit #8f6b498.)
I don't know the reason for this. The code shown by @code_native
for these two functions looks identical,
and the timings for a single call to getindex
and invget
are identical, too.
For the benchmark code see the file benchmark/benchmark_dict.jl
in the repository.