Arrays
is a library to work with well-structured Arrays with fast random-element-access for Elixir, offering a common interface with multiple implementations with varying performance guarantees that can be switched in your configuration.
Arrays is available in Hex and can be installed
by adding arrays
to your list of dependencies in mix.exs
:
def deps do
[
{:arrays, "~> 2.1"}
]
end
Documentation can be found at https://hexdocs.pm/arrays.
By calling Arrays.new
or Arrays.empty
:
iex> Arrays.new(["Dvorak", "Tchaikovsky", "Bruch"])
#Arrays.Implementations.MapArray<["Dvorak", "Tchaikovsky", "Bruch"]>
iex> Arrays.new(["Dvorak", "Tchaikovsky", "Bruch"], implementation: Arrays.Implementations.ErlangArray)
#Arrays.Implementations.ErlangArray<["Dvorak", "Tchaikovsky", "Bruch"]>
By using Collectable
:
iex> [1, 2, 3] |> Enum.into(Arrays.new())
#Arrays.Implementations.MapArray<[1, 2, 3]>
iex> for x <- 1..2, y <- 4..5, into: Arrays.new(), do: {x, y}
#Arrays.Implementations.MapArray<[{1, 4}, {1, 5}, {2, 4}, {2, 5}]>
- Indexing is fast.
- The full Access calls are supported,
- Variants of many common
Enum
-like functions that keep the result an array (rather than turning it into a list), are available.
iex> words = Arrays.new(["the", "quick", "brown", "fox", "jumps", "over", "the", "lazy", "dog"])
#Arrays.Implementations.MapArray<["the", "quick", "brown", "fox", "jumps", "over", "the", "lazy", "dog"]>
iex> Arrays.size(words) # Runs in constant-time
9
iex> words[3] # Indexing is fast
"fox"
iex> words = put_in(words[2], "purple") # All of `Access` is supported
#Arrays.Implementations.MapArray<["the", "quick", "purple", "fox", "jumps", "over", "the", "lazy", "dog"]>
iex> # Common operations are available without having to turn the array back into a list (as `Enum` functions would do):
iex> Arrays.map(words, &String.upcase/1) # Map a function, keep result an array
#Arrays.Implementations.MapArray<["THE", "QUICK", "PURPLE", "FOX", "JUMPS", "OVER", "THE", "LAZY", "DOG"]>
iex> lengths = Arrays.map(words, &String.length/1)
#Arrays.Implementations.MapArray<[3, 5, 6, 3, 5, 4, 3, 4, 3]>
iex> Arrays.reduce(lengths, 0, &Kernel.+/2) # `reduce_right` is supported as well.
36
Concatenating arrays:
iex> Arrays.new([1, 2, 3]) |> Arrays.concat(Arrays.new([4, 5, 6]))
#Arrays.Implementations.MapArray<[1, 2, 3, 4, 5, 6]>
Slicing arrays:
iex> ints = Arrays.new(1..100)
iex> Arrays.slice(ints, 9..19)
#Arrays.Implementations.MapArray<[10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20]>
Algorithms that use arrays can be used while abstracting away from the underlying representation. Which array implementation/representation is actually used, can then later be configured/compared, to make a trade-off between ease-of-use and time/memory efficiency.
Arrays
itself comes with two built-in implementations:
Arrays.Implementations.ErlangArray
wraps the Erlang:array
module, allowing this time-tested implementation to be used with all common Elixir protocols and syntactic sugar.Arrays.Implementations.MapArray
is a simple implementation that uses a map with sequential integers as keys.
By default, the MapArray implementation is used when creating new array objects, but this can be configured by either changing the default in your whole application, or by passing an option to a specific invocation of new/0-2
, or empty/0-1
.
Implementations provided by other libraries:
- ArraysAja adds support for Aja's
A.Vector
, which is an implementation of a 'Hickey Trie' vector. For most operations, it significantly outperformsErlangArray
andMapArray
.
iex> words = Arrays.new(["the", "quick", "brown", "fox", "jumps", "over", "the", "lazy", "dog"])
#Arrays.Implementations.MapArray<["the", "quick", "brown", "fox", "jumps", "over", "the", "lazy", "dog"]>
Besides being able to use all functions in the Arrays
module,
one can use the following protocols and behaviours with them:
-
From Elixir's standard library:
-
Enumerable
: Iterating over arrays -
Collectable
: Creating arrays from collections -
the
Access
behaviour: access a particular element using square brackets,put_in
etc. -
From common container libraries:
-
Insertable
: Append a single item from the end of an array -
Extractable
: Take a single item from the end of an array -
FunLand.Mappable
: Map a function over each element in the array, creating a new array with the results -
FunLand.Reducible
: Reduce an array to a single value.
Note: FunLand
is an optional dependency of this library, so its functionality will only be available if :fun_land
is also added to your mix.exs
dependencies list.
iex> myarray = Arrays.new([2, 1, 4, 2, 0])
iex> Enum.sort(myarray)
[0, 1, 2, 2, 4]
iex> Enum.count(myarray)
5
iex> Enum.with_index(myarray)
[{2, 0}, {1, 1}, {4, 2}, {2, 3}, {0, 4}]
iex> Enum.slice(myarray, 1, 3)
[1, 4, 2]
iex> names = Arrays.new(["Ernie", "Bert", "Kermit"])
iex> names |> Stream.map(&String.upcase/1) |> Enum.into(Arrays.new())
#Arrays.Implementations.MapArray<["ERNIE", "BERT", "KERMIT"]>
iex> foods = Arrays.new(["Cheese", "Strawberries", "Cookies"])
iex> foods |> Enum.take(2)
["Cheese", "Strawberries"]
iex> [10, 20, 30, 40] |> Enum.into(Arrays.new())
#Arrays.Implementations.MapArray<[10, 20, 30, 40]>
iex> arr = Arrays.new([1, 2, 3, 4])
iex> arr = put_in(arr[2], 33)
#Arrays.Implementations.MapArray<[1, 2, 33, 4]>
iex> arr = update_in(arr[1], (&(&1 * -2)))
#Arrays.Implementations.MapArray<[1, -4, 33, 4]>
iex> arr = update_in(arr[-1], (&(&1 + 1)))
#Arrays.Implementations.MapArray<[1, -4, 33, 5]>
iex> {33, arr} = pop_in(arr[-2])
iex> arr
#Arrays.Implementations.MapArray<[1, -4, 5]>
iex> {1, arr} = pop_in(arr[0])
iex> arr
#Arrays.Implementations.MapArray<[-4, 5]>
iex> {5, arr} = pop_in(arr[-1])
iex> arr
#Arrays.Implementations.MapArray<[-4]>
iex> arr2 = Arrays.new([10, 20, 30])
iex> {20, arr2} = get_and_update_in(arr2[1], fn _ -> :pop end)
iex> arr2
#Arrays.Implementations.MapArray<[10, 30]>
square-bracket access, get_in
, put_in
and update_in
are very fast operations.
Unless pop
/pop_in
is used for the last element in the array, is a very slow operation,
as it requires moving of all elements after the given index in the array.
Both positive indexes (counting from zero) and negative indexes
(-1
is the last element, -2
the second-to-last element, etc.) are supported.
However, if positive_index > Arrays.size(array)
or negative_index < -Arrays.size(array)
,
an ArgumentError is raised:
iex> arr = Arrays.new([1,2,3,4])
iex> pop_in(arr[4])
** (ArgumentError) argument error
iex> arr = Arrays.new([1,2,3,4])
iex> pop_in(arr[-5])
** (ArgumentError) argument error
iex> arr = Arrays.new([1,2,3,4])
iex> Access.fetch(arr, 4)
:error
iex> Access.fetch(arr, -5)
:error
iex> arr = Arrays.new([1,2,3,4])
iex> update_in(arr[8], fn x -> x * 2 end)
** (ArgumentError) argument error
iex> arr = Arrays.new([1,2,3,4])
iex> update_in(arr[-8], fn x -> x * 2 end)
** (ArgumentError) argument error
iex> arr = Arrays.new()
iex> {:ok, arr} = Insertable.insert(arr, 42)
iex> {:ok, arr} = Insertable.insert(arr, 100)
iex> arr
#Arrays.Implementations.MapArray<[42, 100]>
iex> Extractable.extract(Arrays.new())
{:error, :empty}
iex> {:ok, {3, arr}} = Extractable.extract(Arrays.new([1, 2, 3]))
iex> arr
#Arrays.Implementations.MapArray<[1, 2]>
Note: FunLand
is an optional dependency of this library.
iex> Arrays.new([1,2,3,4]) |> FunLand.reduce(0, &(&1+&2))
10
iex> Arrays.new([1, 2, 3, 4]) |> FunLand.Mappable.map(fn x -> x * 2 end)
#Arrays.Implementations.MapArray<[2, 4, 6, 8]>
Elixir widely uses List
as default collection type.
Arrays have the folowing differences:
- Arrays keep track of their size. The size of a list needs to be computed.
- Arrays allow fast¹ element indexing. Indexing later elements in a list slows down linearly in the size of the list.
- Pushing a single element to the end of an array is fast¹. Pushing a single element to the end of a list is very slow, taking linear time.
- Pushing a single element to the start of an array is slow, taking linear time. Pushing a single element to the head of a list is fast, taking constant time.
- Appending of arrays takes time proportional to the size of the second array. Appending two lists takes time proportional to the length of the first list. This means that repeated appending
- Lists are allowed to be improper. Arrays can never be improper.
- Many common operations in Elixir transform an enumerable into a list automatically. Arrays are made using
Arrays.new/0
,Arrays.new/1
Arrays.empty/0
, theinto:
option on afor
, orEnum.into
.
¹: Depending on the implementation, 'fast' is either O(1) (constant time, regardless of array size) or O(log(n)) (logarithmic time, becoming a constant amount slower each time the array doubles in size.)
The linear time many operations on lists take, means that the operation becomes twice as slow when the list doubles in size.
To add array-functionality to a custom datastructure, you'll need to implement the Arrays.Protocol
.
Besides these, you probably want to implement the above-mentioned protocols as well.
You can look at the source code of Arrays.CommonProtocolImplementations
for some hints as to how those protocols can be easily implemented, as many functions can be defined as simple wrappers on top of the functions that Arrays.Protocol
itself already provides.
- 2.1.1 Fixes the incorrect typespec for Arrays.map/2 (c.f. PR #95) . Thank you very much, @ulfurinn!
- 2.1.0 Adds the
Arrays.array(element)
type. This type is equivalent toArrays.array()
but indicates (for documentation) what type of elements are expected. Thank you very much, @jzimbel-mbta! - 2.0.0 Improves the
Arrays.Protocol
to be more friendly to implement. Specifically:- Remove implementations for
Access.pop
. Instead, throw an error when people try to use it. - Similarly, throw an error when
:pop
is used insideAccess.get_and_update
- Move
empty
fromAccess.Behaviour
toAccess.Protocol
. - Alter handling of
:default
. It is no longer a required setting, and all arrays are able to work with adefault
passed toresize
. - Related to above: Replace
Arrays.Protocol.resize/2
withArrays.Protocol.resize/3
. (Arrays.resize/2
will call it withnil
as third parameter). size
is no longer a required setting.Arrays.new/2
andArrays.empty/1
have been edited to reflect this.
- Remove implementations for
- 1.2.0 - Adds
ErlangArray.from_raw/1
andErlangArray.to_raw/1
for interop with:array
-records created/consumed by other code. - 1.1.0 - Improved README and general usage examples. Introduces
Arrays.concat/1
,Arrays.concat/2
,Arrays.slice/2
,Arrays.slice/3
. - 1.0.0 - Stable release. Mayor overhaul, 100% test coverage, 100% documentation.
- 0.1.0 - Initial version.
- Add some simple benchmarks
- Appending a single element
- Random element access
- Random element update
- Concatenate two collections
- Add
from_raw
andto_raw
functions to ErlangArray to work with pre-existing code that operates on the raw:array
record itself. - Add more benchmarks:
- Resizing collection (smaller)
- Resizing collection (larger)
- Removing a single element
- Removing all elements one-by-one until the array is empty
- Potentially add more helper functionality to
Arrays
, such assort
,swap
,shuffle
,split
. - Look into adding a persistent bit-partitioned vector trie ('Hickey trie') implementation (potentially based on persistent_vector).
-
Look into adding a NIF-based immutable array implementation, such as im-rs's RRB-Vector, where besides being extra performant because of being close to the metal, having access to the reference-count might allow extra optimizations (in-place mutation is possible when you know that there is only one variable referencing the array). This is a bit of a long shot, but it might be very worthwhile.-> This is done. See ArraysRRBVector. Unfortunately, it turned out to not be more performant because of NIF-calling overhead.
You can run the benchmarks locally by running mix run benchmarks/benchmarks.exs
,
which will also output the HTML format with nice graphs.
From below benchmarks, we know (caveat emptor):
- For collections smaller than ~100-200 elements, there is no pronounced difference between using lists and arrays.
- For collections with more than ~100-200, but fewer than fewer than ~10_000-20_000 elements, ErlangArray is a small constant amount faster than MapArray for updates, and other operations perform similarly.
- For collections with more than ~10_000-20_000 elements, MapArray is usually a small constant amount faster than ErlangArray.
Appending a single element is very fast on arrays, even as sizes grow. MapArray and ErlangArray perform similarly.
For extra comparison, we look at lists both to see how slow list ++ [val]
becomes as baseline,
but also how fast [val | list]
still is:
In certain situations where a list can be treated as 'backwards', this can be a very simple way to append elements. As doing this is built-in, it will always be faster than our arrays. Thus, it serves as a 'maxline'.
Accessing a random element is very fast on arrays, even as sizes grow.
Arrays start beating lists significantly once the collection has more than 256 elements.
MapArray and ErlangArray seem to perform similarly < 8192 elements.
For larger sizes, ErlangArray seems to be a factor ~2 slower than MapArray again.
Arrays start beating lists once the collection has more than 128 elements.
For sizes up to 131072 elements, MapArray seems to be between 100% and 30% slower than ErlangArray. For longer arrays, MapArray wins out, with ErlangArray being ~30% slower.
It seems like put_in
has some overhead w.r.t. calling Arrays.replace
.
This warrants more investigation. Maybe Access
has some overhead for its calls,
or maybe the implementations of get_and_update_in
could be further optimized.
Strangely, concatenation of large collections is very fast on lists. Probably because all of it happens in a single built-in function?
Lists outperform arrays 20x-100x for this task.
Between ErlangArray and MapArray, ErlangArray seems to handle this task 50% faster when concatenating two 4068-element arrays, and twice as fast for larger collections.