List

Resizable array with O(sqrt(n)) memory overhead. Static type List that can be declared stable. For the List class see the file Class.mo.

This implementation is adapted with permission from the vector Mops package created by Research AG.

Copyright: 2023 MR Research AG Main author: Andrii Stepanov Contributors: Timo Hanke (timohanke), Andy Gura (andygura), react0r-com

type List<T> = Types.List<T>

List<T> provides a mutable list of elements of type T. Based on the paper "Resizable Arrays in Optimal Time and Space" by Brodnik, Carlsson, Demaine, Munro and Sedgewick (1999). Since this is internally a two-dimensional array the access times for put and get operations will naturally be 2x slower than Buffer and Array. However, Array is not resizable and Buffer has O(size) memory waste.

public func empty<T>() : List<T>

Creates a new empty List for elements of type T.

Example:


let list = List.new<Nat>(); // Creates a new List

public func singleton<T>(element : T) : List<T>

Returns a new list with capacity and size 1, containing element.

Example:


import Nat "mo:base/Nat";

let list = List.make<Nat>(1);
List.toText<Nat>(list, Nat.toText); // => "[1]"

Runtime: O(1)

Space: O(1)

public func repeat<T>(initValue : T, size : Nat) : List<T>

Create a List with size copies of the initial value.

let list = List.repeat<Nat>(2, 4); // [2, 2, 2, 2]

Runtime: O(size)

public func toPure<T>(list : List<T>) : PureList.List<T>

Converts a mutable List to a purely functional PureList.

Example:

let list = List.fromArray<Nat>([1, 2, 3]);
let pureList = List.toPure<Nat>(list); // converts to immutable PureList

Runtime: O(size)

Space: O(size)

public func fromPure<T>(pure : PureList.List<T>) : List<T>

Converts a purely functional List to a mutable List.

Example:

import PureList "mo:base/pure/List";

let pureList = PureList.fromArray<Nat>([1, 2, 3]);
let list = List.fromPure<Nat>(pureList); // converts to mutable List

Runtime: O(size)

Space: O(size)

public func addRepeat<T>(
  list : List<T>,
  initValue : T,
  count : Nat
)

Add to list count copies of the initial value.

let list = List.repeat<Nat>(2, 4); // [2, 2, 2, 2]
List.addRepeat(list, 2, 1); // [2, 2, 2, 2, 1, 1]

Runtime: O(count)

public func clear<T>(list : List<T>)

Resets the list to size 0, de-referencing all elements.

Example:


List.add(list, 10);
List.add(list, 11);
List.add(list, 12);
List.clear(list); // list is now empty
List.toArray(list) // => []

Runtime: O(1)

public func clone<T>(list : List<T>) : List<T>

Returns a copy of a List, with the same size.

Example:


list.add(1);

let clone = List.clone(list);
List.toArray(clone); // => [1]

Runtime: O(size)

public func map<T, R>(list : List<T>, f : T -> R) : List<R>

Creates and returns a new list, populated with the results of calling a provided function on every element in the provided list

Example:


list.add(1);

let t = List.map<Nat, Text>(list, Nat.toText);
List.toArray(t); // => ["1"]

Runtime: O(size)

public func filter<T>(list : List<T>, predicate : T -> Bool) : List<T>

Returns a new list containing only the elements from list for which the predicate returns true.

Example:

let list = List.fromArray<Nat>([1, 2, 3, 4]);
let evenNumbers = List.filter<Nat>(list, func x = x % 2 == 0);
List.toArray(evenNumbers); // => [2, 4]

Runtime: O(size)

Space: O(size)

*Runtime and space assumes that predicate runs in O(1) time and space.

public func filterMap<T, R>(list : List<T>, f : T -> ?R) : List<R>

Returns a new list containing all elements from list for which the function returns ?element. Discards all elements for which the function returns null.

Example:

let list = List.fromArray<Nat>([1, 2, 3, 4]);
let doubled = List.filterMap<Nat, Nat>(list, func x = if (x % 2 == 0) ?x * 2 else null);
List.toArray(doubled); // => [4, 8]

Runtime: O(size)

Space: O(size)

*Runtime and space assumes that f runs in O(1) time and space.

public func size<T>(list : List<T>) : Nat

Returns the current number of elements in the list.

Example:


List.size(list) // => 0

Runtime: O(1) (with some internal calculations)

public func add<T>(list : List<T>, element : T)

Adds a single element to the end of a List, allocating a new internal data block if needed, and resizing the internal index block if needed.

Example:


List.add(list, 0); // add 0 to list
List.add(list, 1);
List.add(list, 2);
List.add(list, 3);
List.toArray(list) // => [0, 1, 2, 3]

Amortized Runtime: O(1), Worst Case Runtime: O(sqrt(n))

public func removeLast<T>(list : List<T>) : ?T

Removes and returns the last item in the list or null if the list is empty.

Example:


List.add(list, 10);
List.add(list, 11);
List.removeLast(list); // => ?11

Amortized Runtime: O(1), Worst Case Runtime: O(sqrt(n))

Amortized Space: O(1), Worst Case Space: O(sqrt(n))

public func get<T>(list : List<T>, index : Nat) : T

Returns the element at index index. Indexing is zero-based. Traps if index >= size, error message may not be descriptive.

Example:


List.add(list, 10);
List.add(list, 11);
List.get(list, 0); // => 10

Runtime: O(1)

public func getOpt<T>(list : List<T>, index : Nat) : ?T

Returns the element at index index as an option. Returns null when index >= size. Indexing is zero-based.

Example:


List.add(list, 10);
List.add(list, 11);
let x = List.getOpt(list, 0); // => ?10
let y = List.getOpt(list, 2); // => null

Runtime: O(1)

public func put<T>(
  list : List<T>,
  index : Nat,
  value : T
)

Overwrites the current element at index with element. Traps if index >= size. Indexing is zero-based.

Example:


List.add(list, 10);
List.put(list, 0, 20); // overwrites 10 at index 0 with 20
List.toArray(list) // => [20]

Runtime: O(1)

public func sort<T>(list : List<T>, compare : (T, T) -> Order.Order)

Sorts the elements in the list according to compare. Sort is deterministic, stable, and in-place.

Example:


List.add(list, 3);
List.add(list, 1);
List.add(list, 2);
List.sort(list, Nat.compare);
List.toArray(list) // => [1, 2, 3]

Runtime: O(size * log(size))

Space: O(size) *Runtime and space assumes that compare runs in O(1) time and space.

public func indexOf<T>(
  list : List<T>,
  equal : (T, T) -> Bool,
  element : T
) : ?Nat

Finds the first index of element in list using equality of elements defined by equal. Returns null if element is not found.

Example:


let list = List.new<Nat>();
List.add(list, 1);
List.add(list, 2);
List.add(list, 3);
List.add(list, 4);

List.indexOf<Nat>(3, list, Nat.equal); // => ?2

Runtime: O(size)

*Runtime and space assumes that equal runs in O(1) time and space.

public func lastIndexOf<T>(
  list : List<T>,
  equal : (T, T) -> Bool,
  element : T
) : ?Nat

Finds the last index of element in list using equality of elements defined by equal. Returns null if element is not found.

Example:


let list = List.new<Nat>();
List.add(list, 1);
List.add(list, 2);
List.add(list, 3);
List.add(list, 4);
List.add(list, 2);
List.add(list, 2);

List.lastIndexOf<Nat>(2, list, Nat.equal); // => ?5

Runtime: O(size)

*Runtime and space assumes that equal runs in O(1) time and space.

public func firstIndexWhere<T>(list : List<T>, predicate : T -> Bool) : ?Nat

Finds the index of the first element in list for which predicate is true. Returns null if no such element is found.

Example:


let list = List.new<Nat>();
List.add(list, 1);
List.add(list, 2);
List.add(list, 3);
List.add(list, 4);

List.firstIndexWhere<Nat>(list, func(i) { i % 2 == 0 }); // => ?1

Runtime: O(size)

*Runtime and space assumes that predicate runs in O(1) time and space.

public func lastIndexWhere<T>(list : List<T>, predicate : T -> Bool) : ?Nat

Finds the index of the last element in list for which predicate is true. Returns null if no such element is found.

Example:


let list = List.new<Nat>();
List.add(list, 1);
List.add(list, 2);
List.add(list, 3);
List.add(list, 4);

List.lastIndexWhere<Nat>(list, func(i) { i % 2 == 0 }); // => ?3

Runtime: O(size)

*Runtime and space assumes that predicate runs in O(1) time and space.

public func all<T>(list : List<T>, predicate : T -> Bool) : Bool

Returns true iff every element in list satisfies predicate. In particular, if list is empty the function returns true.

Example:


List.add(list, 2);
List.add(list, 3);
List.add(list, 4);

List.all<Nat>(list, func x { x > 1 }); // => true

Runtime: O(size)

Space: O(1)

*Runtime and space assumes that predicate runs in O(1) time and space.

public func any<T>(list : List<T>, predicate : T -> Bool) : Bool

Returns true iff some element in list satisfies predicate. In particular, if list is empty the function returns false.

Example:


List.add(list, 2);
List.add(list, 3);
List.add(list, 4);

List.any<Nat>(list, func x { x > 3 }); // => true

Runtime: O(size)

Space: O(1)

*Runtime and space assumes that predicate runs in O(1) time and space.

public func values<T>(list : List<T>) : Iter.Iter<T>

Returns an Iterator (Iter) over the elements of a List. Iterator provides a single method next(), which returns elements in order, or null when out of elements to iterate over.



List.add(list, 10);
List.add(list, 11);
List.add(list, 12);

var sum = 0;
for (element in List.values(list)) {
  sum += element;
};
sum // => 33

Note: This does not create a snapshot. If the returned iterator is not consumed at once, and instead the consumption of the iterator is interleaved with other operations on the List, then this may lead to unexpected results.

Runtime: O(1)

public func entries<T>(list : List<T>) : Iter.Iter<(T, Nat)>

Returns an Iterator (Iter) over the items, i.e. pairs of value and index of a List. Iterator provides a single method next(), which returns elements in order, or null when out of elements to iterate over.



List.add(list, 10);
List.add(list, 11);
List.add(list, 12);
Iter.toArray(List.entries(list)); // [(10, 0), (11, 1), (12, 2)]

Note: This does not create a snapshot. If the returned iterator is not consumed at once, and instead the consumption of the iterator is interleaved with other operations on the List, then this may lead to unexpected results.

Runtime: O(1)

Warning: Allocates memory on the heap to store ?(T, Nat).

public func valuesRev<T>(list : List<T>) : Iter.Iter<T>

Returns an Iterator (Iter) over the elements of a List in reverse order. Iterator provides a single method next(), which returns elements in reverse order, or null when out of elements to iterate over.



List.add(list, 10);
List.add(list, 11);
List.add(list, 12);

var sum = 0;
for (element in List.valuesRev(list)) {
  sum += element;
};
sum // => 33

Note: This does not create a snapshot. If the returned iterator is not consumed at once, and instead the consumption of the iterator is interleaved with other operations on the List, then this may lead to unexpected results.

Runtime: O(1)

public func entriesRev<T>(list : List<T>) : Iter.Iter<(T, Nat)>

Returns an Iterator (Iter) over the items in reverse order, i.e. pairs of value and index of a List. Iterator provides a single method next(), which returns elements in reverse order, or null when out of elements to iterate over.



List.add(list, 10);
List.add(list, 11);
List.add(list, 12);
Iter.toArray(List.entries(list)); // [(12, 0), (11, 1), (10, 2)]

Note: This does not create a snapshot. If the returned iterator is not consumed at once, and instead the consumption of the iterator is interleaved with other operations on the List, then this may lead to unexpected results.

Runtime: O(1)

Warning: Allocates memory on the heap to store ?(T, Nat).

public func keys<T>(list : List<T>) : Iter.Iter<Nat>

Returns an Iterator (Iter) over the keys (indices) of a List. Iterator provides a single method next(), which returns elements in order, or null when out of elements to iterate over.



List.add(list, 10);
List.add(list, 11);
List.add(list, 12);
Iter.toArray(List.values(list)); // [0, 1, 2]

Note: This does not create a snapshot. If the returned iterator is not consumed at once, and instead the consumption of the iterator is interleaved with other operations on the List, then this may lead to unexpected results.

Runtime: O(1)

public func fromIter<T>(iter : Iter.Iter<T>) : List<T>

Creates a List containing elements from iter.

Example:


import Nat "mo:base/Nat";

let array = [1, 1, 1];
let iter = array.vals();

let list = List.fromIter<Nat>(iter); // => [1, 1, 1]

Runtime: O(size)

public func addAll<T>(list : List<T>, iter : Iter.Iter<T>)

Adds elements to a List from iter.

Example:


import Nat "mo:base/Nat";

let array = [1, 1, 1];
let iter = array.vals();
let list = List.repeat<Nat>(2, 1);

let list = List.addAll<Nat>(list, iter); // => [2, 1, 1, 1]

Runtime: O(size), where n is the size of iter.

public func toArray<T>(list : List<T>) : [T]

Creates an immutable array containing elements from a List.

Example:


List.add(list, 1);
List.add(list, 2);
List.add(list, 3);

List.toArray<Nat>(list); // => [1, 2, 3]

Runtime: O(size)

public func fromArray<T>(array : [T]) : List<T>

Creates a List containing elements from an Array.

Example:


import Nat "mo:base/Nat";

let array = [2, 3];

let list = List.fromArray<Nat>(array); // => [2, 3]

Runtime: O(size)

public func toVarArray<T>(list : List<T>) : [var T]

Creates a mutable Array containing elements from a List.

Example:


List.add(list, 1);
List.add(list, 2);
List.add(list, 3);

List.toVarArray<Nat>(list); // => [1, 2, 3]

Runtime: O(size)

public func fromVarArray<T>(array : [var T]) : List<T>

Creates a List containing elements from a mutable Array.

Example:


import Nat "mo:base/Nat";

let array = [var 2, 3];

let list = List.fromVarArray<Nat>(array); // => [2, 3]

Runtime: O(size)

public func first<T>(list : List<T>) : ?T

Returns the first element of list is empty.

Example:


let list = List.repeat<Nat>(1, 10);

List.first(list); // => ?1

Runtime: O(1)

Space: O(1)

public func last<T>(list : List<T>) : ?T

Returns the last element of list. Traps if list is empty.

Example:


let list = List.fromArray<Nat>([1, 2, 3]);

List.last(list); // => 3

Runtime: O(1)

Space: O(1)

public func forEach<T>(list : List<T>, f : T -> ())

Applies f to each element in list.

Example:


import Nat "mo:base/Nat";
import Debug "mo:base/Debug";

let list = List.fromArray<Nat>([1, 2, 3]);

List.forEach<Nat>(list, func(x) {
  Debug.print(Nat.toText(x)); // prints each element in list
});

Runtime: O(size)

Space: O(size)

*Runtime and space assumes that f runs in O(1) time and space.

public func forEachEntry<T>(list : List<T>, f : (Nat, T) -> ())

Applies f to each item (i, x) in list where i is the key and x is the value.

Example:


import Nat "mo:base/Nat";
import Debug "mo:base/Debug";

let list = List.fromArray<Nat>([1, 2, 3]);

List.forEachEntry<Nat>(list, func (i,x) {
  // prints each item (i,x) in list
  Debug.print(Nat.toText(i) # Nat.toText(x));
});

Runtime: O(size)

Space: O(size)

*Runtime and space assumes that f runs in O(1) time and space.

public func forEachEntryRev<T>(list : List<T>, f : (Nat, T) -> ())

Like forEachEntryRev but iterates through the list in reverse order, from end to beginning.

Example:


import Nat "mo:base/Nat";
import Debug "mo:base/Debug";

let list = List.fromArray<Nat>([1, 2, 3]);

List.forEachEntryRev<Nat>(list, func (i,x) {
  // prints each item (i,x) in list
  Debug.print(Nat.toText(i) # Nat.toText(x));
});

Runtime: O(size)

Space: O(size)

*Runtime and space assumes that f runs in O(1) time and space.

public func forEachRev<T>(list : List<T>, f : T -> ())

Applies f to each element in list in reverse order.

Example:


import Nat "mo:base/Nat";
import Debug "mo:base/Debug";

let list = List.fromArray<Nat>([1, 2, 3]);

List.forEachRev<Nat>(list, func (x) {
  Debug.print(Nat.toText(x)); // prints each element in list in reverse order
});

Runtime: O(size)

Space: O(size)

*Runtime and space assumes that f runs in O(1) time and space.

public func contains<T>(
  list : List<T>,
  equal : (T, T) -> Bool,
  element : T
) : Bool

Returns true if List contains element with respect to equality defined by equal.

Example:


import Nat "mo:base/Nat";

List.add(list, 2);
List.add(list, 0);
List.add(list, 3);

List.contains<Nat>(list, Nat.equal, 2); // => true

Runtime: O(size)

Space: O(1)

*Runtime and space assumes that equal runs in O(1) time and space.

public func max<T>(list : List<T>, compare : (T, T) -> Order.Order) : ?T

Finds the greatest element in list defined by compare. Returns null if list is empty.

Example:


import Nat "mo:base/Nat";

List.add(list, 1);
List.add(list, 2);

List.max<Nat>(list, Nat.compare); // => ?2

Runtime: O(size)

Space: O(1)

*Runtime and space assumes that compare runs in O(1) time and space.

public func min<T>(list : List<T>, compare : (T, T) -> Order.Order) : ?T

Finds the least element in list defined by compare. Returns null if list is empty.

Example:


import Nat "mo:base/Nat";

List.add(list, 1);
List.add(list, 2);

List.min<Nat>(list, Nat.compare); // => ?1

Runtime: O(size)

Space: O(1)

*Runtime and space assumes that compare runs in O(1) time and space.

public func equal<T>(
  list1 : List<T>,
  list2 : List<T>,
  equal : (T, T) -> Bool
) : Bool

Defines equality for two lists, using equal to recursively compare elements in the lists. Returns true iff the two lists are of the same size, and equal evaluates to true for every pair of elements in the two lists of the same index.

Example:


import Nat "mo:base/Nat";

let list1 = List.fromArray<Nat>([1,2]);
let list2 = List.new<Nat>();
list2.add(1);
list2.add(2);

List.equal<Nat>(list1, list2, Nat.equal); // => true

Runtime: O(size)

Space: O(1)

*Runtime and space assumes that equal runs in O(1) time and space.

public func compare<T>(
  list1 : List<T>,
  list2 : List<T>,
  compare : (T, T) -> Order.Order
) : Order.Order

Defines comparison for two lists, using compare to recursively compare elements in the lists. Comparison is defined lexicographically.

Example:


import Nat "mo:base/Nat";

let list1 = List.fromArray<Nat>([1,2]);
let list2 = List.new<Nat>();
list2.add(1);
list2.add(2);

List.compare<Nat>(list1, list2, Nat.compare); // => #less

Runtime: O(size)

Space: O(1)

*Runtime and space assumes that compare runs in O(1) time and space.

public func toText<T>(list : List<T>, f : T -> Text) : Text

Creates a textual representation of list, using toText to recursively convert the elements into Text.

Example:


import Nat "mo:base/Nat";

let list = List.fromArray<Nat>([1,2,3,4]);

List.toText<Nat>(list, Nat.toText); // => "[1, 2, 3, 4]"

Runtime: O(size)

Space: O(size)

*Runtime and space assumes that toText runs in O(1) time and space.

public func foldLeft<A, T>(
  list : List<T>,
  base : A,
  combine : (A, T) -> A
) : A

Collapses the elements in list into a single value by starting with base and progessively combining elements into base with combine. Iteration runs left to right.

Example:


import Nat "mo:base/Nat";

let list = List.fromArray<Nat>([1,2,3]);

List.foldLeft<Text, Nat>(list, "", func (acc, x) { acc # Nat.toText(x)}); // => "123"

Runtime: O(size)

Space: O(1)

*Runtime and space assumes that combine runs in O(1)` time and space.

public func foldRight<T, A>(
  list : List<T>,
  base : A,
  combine : (T, A) -> A
) : A

Collapses the elements in list into a single value by starting with base and progessively combining elements into base with combine. Iteration runs right to left.

Example:


import Nat "mo:base/Nat";

let list = List.fromArray<Nat>([1,2,3]);

List.foldRight<Nat, Text>(list, "", func (x, acc) { Nat.toText(x) # acc }); // => "123"

Runtime: O(size)

Space: O(1)

*Runtime and space assumes that combine runs in O(1)` time and space.

public func reverseInPlace<T>(list : List<T>)

Reverses the order of elements in list by overwriting in place.

Example:


import Nat "mo:base/Nat";

let list = List.fromArray<Nat>([1,2,3]);

List.reverseInPlace<Nat>(list);
List.toText<Nat>(list, Nat.toText); // => "[3, 2, 1]"

Runtime: O(size)

Space: O(1)

public func reverse<T>(list : List<T>) : List<T>

Returns a new List with the elements from list in reverse order.

Example:


import Nat "mo:base/Nat";

let list = List.fromArray<Nat>([1,2,3]);

let rlist = List.reverse<Nat>(list);
List.toText<Nat>(rlist, Nat.toText); // => "[3, 2, 1]"

Runtime: O(size)

Space: O(1)

public func isEmpty<T>(list : List<T>) : Bool

Returns true if and only if the list is empty.

Example:


let list = List.fromArray<Nat>([2,0,3]);
List.isEmpty<Nat>(list); // => false

Runtime: O(1)

Space: O(1)