base/OrderedSet
Stable ordered set implemented as a red-black tree.
A red-black tree is a balanced binary search tree ordered by the elements.
The tree data structure internally colors each of its nodes either red or black, and uses this information to balance the tree during the modifying operations.
| Runtime | Space | 
|---|---|
| O(log(n))(worst case per insertion, removal, or retrieval) | O(n)(for storing the entire tree) | 
n denotes the number of key-value entries (i.e. nodes) stored in the tree.
Unless stated otherwise, operations that iterate over or modify the map (such as insertion, deletion, traversal, and transformation) may create temporary objects with worst-case space usage of O(log(n)) or O(n). These objects are short-lived and will be collected by the garbage collector automatically.
Runtime and space complexity assumes that compare, equal, and other functions execute in O(1) time and space.
The core of this implementation is derived from:
- Ken Friis Larsen's RedBlackMap.sml, which itself is based on:
- Stefan Kahrs, "Red-black trees with types", Journal of Functional Programming, 11(4): 425-432 (2001), version 1 in web appendix.
Type Set
type Set<T> = { size : Nat; root : Tree<T> }
Ordered collection of unique elements of the generic type T.
If type T is stable then Set<T> is also stable.
To ensure that property the Set<T> does not have any methods,
instead they are gathered in the functor-like class Operations (see example there).
Class Operations<T>
class Operations<T>(compare : (T, T) -> O.Order)
Class that captures element type T along with its ordering function compare
and provide all operations to work with a set of type Set<T>.
An instance object should be created once as a canister field to ensure that the same ordering function is used for every operation.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
actor {
  let natSet = Set.Make<Nat>(Nat.compare); // : Operations<Nat>
  stable var usedIds : Set.Set<Nat> = natSet.empty();
  public func createId(id : Nat) : async () {
    usedIds := natSet.put(usedIds, id);
  };
  public func idIsUsed(id: Nat) : async Bool {
     natSet.contains(usedIds, id)
  }
}
Function fromIter
func fromIter(i : I.Iter<T>) : Set<T>
Returns a new Set, containing all entries given by the iterator i.
If there are multiple identical entries only one is taken.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
Debug.print(debug_show(Iter.toArray(natSet.vals(set))));
// [0, 1, 2]
| Runtime | Space | 
|---|---|
| O(n * log(n)) | O(n)(retained memory + garbage) | 
Function put
func put(s : Set<T>, value : T) : Set<T>
Insert the value value into the set s. Has no effect if value is already
present in the set. Returns a modified set.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
var set = natSet.empty();
set := natSet.put(set, 0);
set := natSet.put(set, 2);
set := natSet.put(set, 1);
Debug.print(debug_show(Iter.toArray(natSet.vals(set))));
// [0, 1, 2]
Runtime: O(log(n)).
Space: O(log(n)).
where n denotes the number of elements stored in the set and
assuming that the compare function implements an O(1) comparison.
Note: The returned set shares with the s most of the tree nodes.
Garbage collecting one of sets (e.g. after an assignment m := natSet.delete(m, k))
causes collecting O(log(n)) nodes.
Function delete
func delete(s : Set<T>, value : T) : Set<T>
Deletes the value value from the set s. Has no effect if value is not
present in the set. Returns modified set.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
Debug.print(debug_show(Iter.toArray(natSet.vals(natSet.delete(set, 1)))));
Debug.print(debug_show(Iter.toArray(natSet.vals(natSet.delete(set, 42)))));
// [0, 2]
// [0, 1, 2]
| Runtime | Space | 
|---|---|
| O(log(n)) | O(log(n)) | 
Function contains
func contains(s : Set<T>, value : T) : Bool
Test if the set 's' contains a given element.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
Debug.print(debug_show natSet.contains(set, 1)); // => true
Debug.print(debug_show natSet.contains(set, 42)); // => false
| Runtime | Space | 
|---|---|
| O(log(n)) | O(1) | 
Function max
func max(s : Set<T>) : ?T
Get a maximal element of the set s if it is not empty, otherwise returns null
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let s1 = natSet.fromIter(Iter.fromArray([0, 2, 1]));
let s2 = natSet.empty();
Debug.print(debug_show(natSet.max(s1))); // => ?2
Debug.print(debug_show(natSet.max(s2))); // => null
| Runtime | Space | 
|---|---|
| O(log(n)) | O(1) | 
Function min
func min(s : Set<T>) : ?T
Get a minimal element of the set s if it is not empty, otherwise returns null
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let s1 = natSet.fromIter(Iter.fromArray([0, 2, 1]));
let s2 = natSet.empty();
Debug.print(debug_show(natSet.min(s1))); // => ?0
Debug.print(debug_show(natSet.min(s2))); // => null
| Runtime | Space | 
|---|---|
| O(log(n)) | O(log(1)) | 
Function union
func union(s1 : Set<T>, s2 : Set<T>) : Set<T>
Set union operation.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set1 = natSet.fromIter(Iter.fromArray([0, 1, 2]));
let set2 = natSet.fromIter(Iter.fromArray([2, 3, 4]));
Debug.print(debug_show Iter.toArray(natSet.vals(natSet.union(set1, set2))));
// [0, 1, 2, 3, 4]
| Runtime | Space | 
|---|---|
| O(m* log(n)) | O(m)retained + garbage | 
Function intersect
func intersect(s1 : Set<T>, s2 : Set<T>) : Set<T>
Set intersection operation.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set1 = natSet.fromIter(Iter.fromArray([0, 1, 2]));
let set2 = natSet.fromIter(Iter.fromArray([1, 2, 3]));
Debug.print(debug_show Iter.toArray(natSet.vals(natSet.intersect(set1, set2))));
// [1, 2]
| Runtime | Space | 
|---|---|
| O(m* log(n)) | O(m)retained + garbage | 
Note: Creates O(m) temporary objects that will be collected as garbage.
Function diff
func diff(s1 : Set<T>, s2 : Set<T>) : Set<T>
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set1 = natSet.fromIter(Iter.fromArray([0, 1, 2]));
let set2 = natSet.fromIter(Iter.fromArray([1, 2, 3]));
Debug.print(debug_show Iter.toArray(natSet.vals(natSet.diff(set1, set2))));
// [0]
| Runtime | Space | 
|---|---|
| O(m* log(n)) | O(m)retained + garbage | 
Function map
func map<T1>(s : Set<T1>, f : T1 -> T) : Set<T>
Creates a new Set by applying f to each entry in the set s. Each element
x in the old set is transformed into a new entry x2, where
the new value x2 is created by applying f to x.
The result set may be smaller than the original set due to duplicate elements.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 1, 2, 3]));
func f(x : Nat) : Nat = if (x < 2) { x } else { 0 };
let resSet = natSet.map(set, f);
Debug.print(debug_show(Iter.toArray(natSet.vals(resSet))));
// [0, 1]
Cost of mapping all the elements:
| Runtime     | Space         |
|-------------|---------------|
| O(n* log(n)) | O(n)retained + garbage   |
Function mapFilter
func mapFilter<T1>(s : Set<T1>, f : T1 -> ?T) : Set<T>
Creates a new set by applying f to each element in the set s. For each element
x in the old set, if f evaluates to null, the element is discarded.
Otherwise, the entry is transformed into a new entry x2, where
the new value x2 is the result of applying f to x.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 1, 2, 3]));
func f(x : Nat) : ?Nat {
  if(x == 0) {null}
  else { ?( x * 2 )}
};
let newRbSet = natSet.mapFilter(set, f);
Debug.print(debug_show(Iter.toArray(natSet.vals(newRbSet))));
// [2, 4, 6]
| Runtime | Space | 
|---|---|
| O(n* log(n)) | O(n)retained + garbage | 
Function isSubset
func isSubset(s1 : Set<T>, s2 : Set<T>) : Bool
Test if set1 is subset of set2.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set1 = natSet.fromIter(Iter.fromArray([1, 2]));
let set2 = natSet.fromIter(Iter.fromArray([0, 2, 1]));
Debug.print(debug_show natSet.isSubset(set1, set2)); // => true
| Runtime | Space | 
|---|---|
| O(n* log(n)) | O(1) | 
Function equals
func equals(s1 : Set<T>, s2 : Set<T>) : Bool
Test if two sets are equal.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set1 = natSet.fromIter(Iter.fromArray([0, 2, 1]));
let set2 = natSet.fromIter(Iter.fromArray([1, 2]));
Debug.print(debug_show natSet.equals(set1, set1)); // => true
Debug.print(debug_show natSet.equals(set1, set2)); // => false
| Runtime | Space | 
|---|---|
| O(m * log(n)) | O(1) | 
Function vals
func vals(s : Set<T>) : I.Iter<T>
Returns an Iterator (Iter) over the elements of the set.
Iterator provides a single method next(), which returns
elements in ascending order, or null when out of elements to iterate over.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
Debug.print(debug_show(Iter.toArray(natSet.vals(set))));
// [0, 1, 2]
| Runtime | Space | 
|---|---|
| O(n) | O(log(n))retained + garbage | 
Function valsRev
func valsRev(s : Set<T>) : I.Iter<T>
Same as vals() but iterates over elements of the set s in the descending order.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
Debug.print(debug_show(Iter.toArray(natSet.valsRev(set))));
// [2, 1, 0]
| Runtime | Space | 
|---|---|
| O(n) | O(log(n))retained + garbage | 
Function empty
func empty() : Set<T>
Create a new empty Set.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.empty();
Debug.print(debug_show(natSet.size(set))); // => 0
Cost of empty set creation
Runtime: O(1).
Space: O(1)
Function size
func size(s : Set<T>) : Nat
Returns the number of elements in the set.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
Debug.print(debug_show(natSet.size(set))); // => 3
| Runtime | Space | 
|---|---|
| O(1) | O(1) | 
Function foldLeft
func foldLeft<Accum>(set : Set<T>, base : Accum, combine : (Accum, T) -> Accum) : Accum
Collapses the elements in set into a single value by starting with base
and progessively combining elements into base with combine. Iteration runs
left to right.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
func folder(accum : Nat, val : Nat) : Nat = val + accum;
Debug.print(debug_show(natSet.foldLeft(set, 0, folder)));
// 3
| Runtime | Space | 
|---|---|
| O(n) | Depends on combine+O(n)garbage | 
Function foldRight
func foldRight<Accum>(set : Set<T>, base : Accum, combine : (T, Accum) -> Accum) : Accum
Collapses the elements in set into a single value by starting with base
and progessively combining elements into base with combine. Iteration runs
right to left.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
func folder(val : Nat, accum : Nat) : Nat = val + accum;
Debug.print(debug_show(natSet.foldRight(set, 0, folder)));
// 3
| Runtime | Space | 
|---|---|
| O(n) | Depends on combine+O(n)garbage | 
Function isEmpty
func isEmpty(s : Set<T>) : Bool
Test if the given set s is empty.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.empty();
Debug.print(debug_show(natSet.isEmpty(set))); // => true
Runtime: O(1).
Space: O(1).
Function all
func all(s : Set<T>, pred : T -> Bool) : Bool
Test whether all values in the set s satisfy a given predicate pred.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
Debug.print(debug_show(natSet.all(set, func (v) = (v < 10))));
// true
Debug.print(debug_show(natSet.all(set, func (v) = (v < 2))));
// false
| Runtime | Space | 
|---|---|
| O(n) | O(n) | 
Function some
func some(s : Set<T>, pred : (T) -> Bool) : Bool
Test if there exists an element in the set s satisfying the given predicate pred.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
import Iter "mo:base/Iter";
import Debug "mo:base/Debug";
let natSet = Set.Make<Nat>(Nat.compare);
let set = natSet.fromIter(Iter.fromArray([0, 2, 1]));
Debug.print(debug_show(natSet.some(set, func (v) = (v >= 3))));
// false
Debug.print(debug_show(natSet.some(set, func (v) = (v >= 0))));
// true
| Runtime | Space | 
|---|---|
| O(n) | O(1) | 
Function validate
func validate(s : Set<T>) : ()
Test helper that check internal invariant for the given set s.
Raise an error (for a stack trace) if invariants are violated.
Function Make
func Make<T>(compare : (T, T) -> O.Order) : Operations<T>
Create OrderedSet.Operations object capturing element type T and compare function.
It is an alias for the Operations constructor.
Example:
import Set "mo:base/OrderedSet";
import Nat "mo:base/Nat";
actor {
  let natSet = Set.Make<Nat>(Nat.compare);
  stable var set : Set.Set<Nat> = natSet.empty();
};