core/pure/Queue
Double-ended queue of a generic element type T.
The interface is purely functional, not imperative, and queues are immutable values. In particular, Queue operations such as push and pop do not update their input queue but, instead, return the value of the modified Queue, alongside any other data. The input queue is left unchanged.
Examples of use-cases:
Queue (FIFO) by using pushBack() and popFront().
Stack (LIFO) by using pushFront() and popFront().
A Queue is internally implemented as two lists, a head access list and a (reversed) tail access list, that are dynamically size-balanced by splitting.
Construction: Create a new queue with the empty<T>() function.
Note on the costs of push and pop functions:
- Runtime: O(1)amortized costs,O(size)worst case cost per single call.
- Space: O(1)amortized costs,O(size)worst case cost per single call.
n denotes the number of elements stored in the queue.
Note that some operations that traverse the elements of the queue (e.g. forEach, values) preserve the order of the elements,
whereas others (e.g. map, contains) do NOT guarantee that the elements are visited in any order.
The order is undefined to avoid allocations, making these operations more efficient.
import Queue "mo:core/pure/Queue";
Type Queue
type Queue<T> = Types.Pure.Queue<T>
Double-ended queue data type.
Function empty
func empty<T>() : Queue<T>
Create a new empty queue.
Example:
persistent actor {
  let queue = Queue.empty<Nat>();
  assert Queue.isEmpty(queue);
}
Runtime: O(1).
Space: O(1).
Function isEmpty
func isEmpty<T>(queue : Queue<T>) : Bool
Determine whether a queue is empty.
Returns true if queue is empty, otherwise false.
Example:
persistent actor {
  let queue = Queue.empty<Nat>();
  assert Queue.isEmpty(queue);
}
Runtime: O(1).
Space: O(1).
Function singleton
func singleton<T>(item : T) : Queue<T>
Create a new queue comprising a single element.
Example:
persistent actor {
  let queue = Queue.singleton(25);
  assert Queue.size(queue) == 1;
}
Runtime: O(1).
Space: O(1).
Function size
func size<T>(queue : Queue<T>) : Nat
Determine the number of elements contained in a queue.
Example:
persistent actor {
  let queue = Queue.singleton(42);
  assert Queue.size(queue) == 1;
}
Runtime: O(1) in Release profile (compiled with --release flag), O(size) otherwise.
Space: O(1).
Function contains
func contains<T>(queue : Queue<T>, equal : (T, T) -> Bool, item : T) : Bool
Check if a queue contains a specific element.
Returns true if the queue contains an element equal to item according to the equal function.
Note: The order in which elements are visited is undefined, for performance reasons.
Example:
import Nat "mo:core/Nat";
persistent actor {
  let queue = Queue.fromIter([1, 2, 3].values());
  assert Queue.contains(queue, Nat.equal, 2);
  assert not Queue.contains(queue, Nat.equal, 4);
}
Runtime: O(size)
Space: O(1)
Function peekFront
func peekFront<T>(queue : Queue<T>) : ?T
Inspect the optional element on the front end of a queue.
Returns null if queue is empty. Otherwise, the front element of queue.
Example:
persistent actor {
  let queue = Queue.pushFront(Queue.pushFront(Queue.empty(), 2), 1);
  assert Queue.peekFront(queue) == ?1;
}
Runtime: O(1).
Space: O(1).
Function peekBack
func peekBack<T>(queue : Queue<T>) : ?T
Inspect the optional element on the back end of a queue.
Returns null if queue is empty. Otherwise, the back element of queue.
Example:
persistent actor {
  let queue = Queue.pushBack(Queue.pushBack(Queue.empty(), 1), 2);
  assert Queue.peekBack(queue) == ?2;
}
Runtime: O(1).
Space: O(1).
Function pushFront
func pushFront<T>(queue : Queue<T>, element : T) : Queue<T>
Insert a new element on the front end of a queue.
Returns the new queue with element in the front followed by the elements of queue.
This may involve dynamic rebalancing of the two, internally used lists.
Example:
persistent actor {
  let queue = Queue.pushFront(Queue.pushFront(Queue.empty(), 2), 1);
  assert Queue.peekFront(queue) == ?1;
  assert Queue.peekBack(queue) == ?2;
  assert Queue.size(queue) == 2;
}
Runtime: O(size) worst-case, amortized to O(1).
Space: O(size) worst-case, amortized to O(1).
n denotes the number of elements stored in the queue.
Function pushBack
func pushBack<T>(queue : Queue<T>, element : T) : Queue<T>
Insert a new element on the back end of a queue.
Returns the new queue with all the elements of queue, followed by element on the back.
This may involve dynamic rebalancing of the two, internally used lists.
Example:
persistent actor {
  let queue = Queue.pushBack(Queue.pushBack(Queue.empty(), 1), 2);
  assert Queue.peekBack(queue) == ?2;
  assert Queue.size(queue) == 2;
}
Runtime: O(size) worst-case, amortized to O(1).
Space: O(size) worst-case, amortized to O(1).
n denotes the number of elements stored in the queue.
Function popFront
func popFront<T>(queue : Queue<T>) : ?(T, Queue<T>)
Remove the element on the front end of a queue.
Returns null if queue is empty. Otherwise, it returns a pair of
the first element and a new queue that contains all the remaining elements of queue.
This may involve dynamic rebalancing of the two, internally used lists.
Example:
import Runtime "mo:core/Runtime";
persistent actor {
  let initial = Queue.pushBack(Queue.pushBack(Queue.empty(), 1), 2);
  // initial queue with elements [1, 2]
  switch (Queue.popFront(initial)) {
    case null Runtime.trap "Empty queue impossible";
    case (?(frontElement, remainingQueue)) {
      assert frontElement == 1;
      assert Queue.size(remainingQueue) == 1
    }
  }
}
Runtime: O(size) worst-case, amortized to O(1).
Space: O(size) worst-case, amortized to O(1).
n denotes the number of elements stored in the queue.
Function popBack
func popBack<T>(queue : Queue<T>) : ?(Queue<T>, T)
Remove the element on the back end of a queue.
Returns null if queue is empty. Otherwise, it returns a pair of
a new queue that contains the remaining elements of queue
and, as the second pair item, the removed back element.
This may involve dynamic rebalancing of the two, internally used lists.
Example:
import Runtime "mo:core/Runtime";
persistent actor {
  let initial = Queue.pushBack(Queue.pushBack(Queue.empty(), 1), 2);
  // initial queue with elements [1, 2]
  let reduced = Queue.popBack(initial);
  switch reduced {
    case null Runtime.trap("Empty queue impossible");
    case (?result) {
      let reducedQueue = result.0;
      let removedElement = result.1;
      assert removedElement == 2;
      assert Queue.size(reducedQueue) == 1;
    }
  }
}
Runtime: O(size) worst-case, amortized to O(1).
Space: O(size) worst-case, amortized to O(1).
n denotes the number of elements stored in the queue.
Function fromIter
func fromIter<T>(iter : Iter.Iter<T>) : Queue<T>
Turn an iterator into a queue, consuming it. Example:
persistent actor {
  let queue = Queue.fromIter([0, 1, 2, 3, 4].values());
  assert Queue.size(queue) == 5;
}
Runtime: O(size)
Space: O(size)
Function values
func values<T>(queue : Queue<T>) : Iter.Iter<T>
Convert a queue to an iterator of its elements in front-to-back order.
Performance note: Creating the iterator needs O(size) runtime and space!
Example:
import Iter "mo:core/Iter";
persistent actor {
  let queue = Queue.fromIter([1, 2, 3].values());
  assert Iter.toArray(Queue.values(queue)) == [1, 2, 3];
}
Runtime: O(size)
Space: O(size)
Function equal
func equal<T>(queue1 : Queue<T>, queue2 : Queue<T>, equal : (T, T) -> Bool) : Bool
Compare two queues for equality using the provided equality function.
Example:
import Nat "mo:core/Nat";
persistent actor {
  let queue1 = Queue.fromIter([1, 2].values());
  let queue2 = Queue.fromIter([1, 2].values());
  let queue3 = Queue.fromIter([1, 3].values());
  assert Queue.equal(queue1, queue2, Nat.equal);
  assert not Queue.equal(queue1, queue3, Nat.equal);
}
Runtime: O(size)
Space: O(size)
Function all
func all<T>(queue : Queue<T>, predicate : T -> Bool) : Bool
Return true if the given predicate f is true for all queue
elements.
Example:
persistent actor {
  let queue = Queue.fromIter([1, 2, 3].values());
  let allGreaterThanOne = Queue.all<Nat>(queue, func n = n > 1);
  assert not allGreaterThanOne; // false because 1 is not > 1
}
Runtime: O(size)
Space: O(size) as the current implementation uses values to iterate over the queue.
*Runtime and space assumes that the predicate runs in O(1) time and space.
Function any
func any<T>(queue : Queue<T>, predicate : T -> Bool) : Bool
Return true if there exists a queue element for which
the given predicate f is true.
Example:
persistent actor {
  let queue = Queue.fromIter([1, 2, 3].values());
  let hasGreaterThanOne = Queue.any<Nat>(queue, func n = n > 1);
  assert hasGreaterThanOne; // true because 2 and 3 are > 1
}
Runtime: O(size)
Space: O(size) as the current implementation uses values to iterate over the queue.
*Runtime and space assumes that the predicate runs in O(1) time and space.
Function forEach
func forEach<T>(queue : Queue<T>, f : T -> ())
Call the given function for its side effect, with each queue element in turn. The order of visiting elements is front-to-back.
Example:
persistent actor {
  var text = "";
  let queue = Queue.fromIter(["A", "B", "C"].values());
  Queue.forEach<Text>(queue, func n = text #= n);
  assert text == "ABC";
}
Runtime: O(size)
Space: O(size)
*Runtime and space assumes that f runs in O(1) time and space.
Function map
func map<T1, T2>(queue : Queue<T1>, f : T1 -> T2) : Queue<T2>
Call the given function f on each queue element and collect the results
in a new queue.
Note: The order of visiting elements is undefined with the current implementation.
Example:
import Iter "mo:core/Iter";
import Nat "mo:core/Nat";
persistent actor {
  let queue = Queue.fromIter([0, 1, 2].values());
  let textQueue = Queue.map<Nat, Text>(queue, Nat.toText);
  assert Iter.toArray(Queue.values(textQueue)) == ["0", "1", "2"];
}
Runtime: O(size)
Space: O(size)
*Runtime and space assumes that f runs in O(1) time and space.
Function filter
func filter<T>(queue : Queue<T>, predicate : T -> Bool) : Queue<T>
Create a new queue with only those elements of the original queue for which the given function (often called the predicate) returns true.
Note: The order of visiting elements is undefined with the current implementation.
Example:
persistent actor {
  let queue = Queue.fromIter([0, 1, 2, 1].values());
  let filtered = Queue.filter<Nat>(queue, func n = n != 1);
  assert Queue.size(filtered) == 2;
}
Runtime: O(size)
Space: O(size)
*Runtime and space assumes that predicate runs in O(1) time and space.
Function filterMap
func filterMap<T, U>(queue : Queue<T>, f : T -> ?U) : Queue<U>
Call the given function on each queue element, and collect the non-null results in a new queue.
Note: The order of visiting elements is undefined with the current implementation.
Example:
persistent actor {
  let queue = Queue.fromIter([1, 2, 3].values());
  let doubled = Queue.filterMap<Nat, Nat>(
    queue,
    func n = if (n > 1) ?(n * 2) else null
  );
  assert Queue.size(doubled) == 2;
}
Runtime: O(size)
Space: O(size)
*Runtime and space assumes that f runs in O(1) time and space.
Function toText
func toText<T>(queue : Queue<T>, f : T -> Text) : Text
Convert a queue to its text representation using the provided conversion function. This function is meant to be used for debugging and testing purposes.
Example:
import Nat "mo:core/Nat";
persistent actor {
  let queue = Queue.fromIter([1, 2, 3].values());
  assert Queue.toText(queue, Nat.toText) == "PureQueue[1, 2, 3]";
}
Runtime: O(size)
Space: O(size)
Function compare
func compare<T>(queue1 : Queue<T>, queue2 : Queue<T>, compareItem : (T, T) -> Order.Order) : Order.Order
Compare two queues using lexicographic ordering specified by argument function compareItem.
Example:
import Nat "mo:core/Nat";
persistent actor {
  let queue1 = Queue.fromIter([1, 2].values());
  let queue2 = Queue.fromIter([1, 3].values());
  assert Queue.compare(queue1, queue2, Nat.compare) == #less;
}
Runtime: O(size)
Space: O(size)
*Runtime and space assumes that argument compareItem runs in O(1) time and space.
Function reverse
func reverse<T>(queue : Queue<T>) : Queue<T>
Reverse the order of elements in a queue. This operation is cheap, it does NOT require copying the elements.
Example:
persistent actor {
  let queue = Queue.fromIter([1, 2, 3].values());
  let reversed = Queue.reverse(queue);
  assert Queue.peekFront(reversed) == ?3;
  assert Queue.peekBack(reversed) == ?1;
}
Runtime: O(1)
Space: O(1)