On this page:
Set
Set.of
Set.later_  of
Readable  Set
Mutable  Set
Weak  Mutable  Set
Mutable  Set.now_  of
Mutable  Set.later_  of
Set.by
Mutable  Set.by
Weak  Mutable  Set.by
Set
Set
Set.by
Set.by
Set
Readable  Set
Set.by
Set
Set.by
Mutable  Set
Mutable  Set.by
Weak  Mutable  Set
Weak  Mutable  Set.by
Set.empty
Set.empty
Readable  Set.empty
Readable  Set.empty
Set.length
Set.get
Set.has_  element
Set.append
Set.union
Set.intersect
Set.remove
Mutable  Set.set
Mutable  Set.delete
Set.to_  list
Set.copy
Set.snapshot
Set.to_  sequence
8.15.0.12

8.6 Sets🔗ℹ

Immutable sets can be constructed using the syntax {val_expr, ...}, which creates a set containing the values of the val_exprs. More precisely, a use of curly braces with no preceding expression is parsed as an implicit use of the #%braces form.

A set is indexable, and [] after a set expression with an expression for a value checks for membership: the result is a boolean indicating whether the value is in the set. Mutable sets can be updated with a combination of [] and the := operator, where a #false result on the right-hand side of := removes an element from a set, and any other right-hand side result causes the value to be included in the set. (Use ++ to functionally add to an immutable set.) These uses of [] are implemented by #%index. A set can be used as sequence, in which case it supplies its elements in an unspecified order.

annotation

Set

 

annotation

Set.of(annot)

 

annotation

Set.later_of(annot)

 

annotation

ReadableSet

 

annotation

MutableSet

 

annotation

WeakMutableSet

 

annotation

MutableSet.now_of(annot)

 

annotation

MutableSet.later_of(annot)

 

annotation

Set.by(key_comp)

 

annotation

Set.by(key_comp).of(annot)

 

annotation

MutableSet.by(key_comp)

 

annotation

WeakMutableSet.by(key_comp)

The Set annotation matches any set. ReadableSet matches both mutable and immutable sets, while MutableSet matches mutable sets (created with, for example, the MutableSet constructor). WeakMutableSet matches weak mutable sets (created with, for example, the WeakMutableSet constructor).

The of and now_of annotation variants match a set whose elements satisfy annot, where satisfaction of the annotation is confirmed by immediately checking all elements. No future obligation is attached to a set satisfying the annotation, so in the case of MutableSet.now_of, no static information is associated with value access using [].

The later_of annotation variants create a converter annotation given an annotations for elements; satisfaction of those annotations is confirmed only on demand, both for elements that are extracted from the set and for elements added or appended to the set. For Set.later_of, the key and value annotations must be predicate annotations. Since an element annotation is checked on every access, its static information is associated with access using [].

The Set.by and MutableSet.by annotation variants match only sets that use the hash and equality functions specified by key_comp.

Static information associated by Set, etc., makes an expression acceptable to for in static mode.

expression

Set{expr_or_splice, ...}

 

repetition

Set{repet_or_splice, ...}

 

expr_or_splice

 = 

expr

 | 

repet , ellipses

 | 

& set_expr

 

ellipses

 = 

ellipsis

 | 

ellipses , ellipsis

 

ellipsis

 = 

...

 

function

fun Set(val :: Any, ...) :: Set

 

expression

Set.by(key_comp){expr_or_splice, ...}

 

expression

Set.by(key_comp)

 

repetition

Set.by(key_comp){repet_or_splice, ...}

Constructs an immutable set containing given values, equivalent to using {expr_or_splice, ...} to form a set (see #%braces).

Note that Set{} and Set() produce an empty set while {} does not, since {} produces an empty map instead.

The Set.by variants create a set that uses the equality and hashing functions specified by key_comp.

> def s = Set{"x", 1, "y", 2}

> s

{1, 2, "x", "y"}

> s["x"]

#true

> s[1]

#true

> s[42]

#false

> Set("x", 1, "y", 2)

{1, 2, "x", "y"}

binding operator

Set{expr, ...}

 

binding operator

Set{expr, ..., rest}

 

binding operator

Set(expr, ...)

 

binding operator

Set(expr, ..., rest)

 

binding operator

ReadableSet{expr, ...}

 

binding operator

ReadableSet{expr, ..., rest}

 

binding operator

ReadableSet(expr, ...)

 

binding operator

ReadableSet(expr, ..., rest)

 

binding operator

Set.by(key_comp){expr, ...}

 

binding operator

Set.by(key_comp){expr, ..., rest}

 

binding operator

Set.by(key_comp)(expr, ...)

 

binding operator

Set.by(key_comp)(expr, ..., rest)

 

rest

 = 

& set_bind

 | 

rest_bind , ellipsis

 

ellipsis

 = 

...

Matches a set containing at least the values computed by the exprs. The matched set may have additional values. If & set_bind is supplied, the rest of the set excluding the values of the given exprs must match the set_bind. Static information associated by Set is propagated to set_bind. If rest_bind followed by ... is supplied, the rest of the set excluding the given exprs must have individual values that match rest_bind, and identifiers in rest_bind are bound as repetitions. Values matching rest_bind are extracted eagerly and preserved in an internal list to implement the repetition.

The Set binding forms match only immutable sets, while ReadableSet forms match both immutable and mutable sets. For ReadableSet, the & set_bind will match a snapshot (in the sense of Set.snapshot) of the rest of the set. The Set.by binding forms match only immutable sets constructed using key_comp.

> def Set{"x", "y"} = {"x", "y"}

> def Set{"x", "y"} = {"x"}

def: value does not satisfy annotation

  value: {"x"}

  annotation: matching(Set{"x", "y""x", "y"})

> def Set{"a"} = {"a", "b"}

> def Set{"a", & rst} = {"a", "b", "c"}

> rst

{"b", "c"}

> def Set{"a", val, ...} = {"a", "b", "c"}

> [val, ...]

["c", "b"]

reducer

Set

 

reducer

Set.by(key_comp)

A reducer used with for, accumulates values into a set.

The Set.by reducer creates a set that uses the equality and hashing functions specified by key_comp.

expression

MutableSet{val_expr, ...}

 

function

fun MutableSet(val :: Any, ...) :: MutableSet

 

expression

MutableSet.by(key_comp){val_expr, ...}

 

expression

MutableSet.by(key_comp)

Similar to Set as a constructor, but creates a mutable set that can be updated using an assignment operator like :=.

Note that ... and & are not supported for construction mutable sets, only immutable sets.

> def m = MutableSet{"x", 1, "y", 2}

> m

MutableSet{1, 2, "x", "y"}

> m["x"]

#true

> m["x"] := #false

> m

MutableSet{1, 2, "y"}

> m["x"] := #true

> m

MutableSet{1, 2, "x", "y"}

expression

WeakMutableSet{val_expr, ...}

 

function

fun WeakMutableSet(val :: Any, ...) :: MutableSet

 

expression

WeakMutableSet.by(key_comp){val_expr, ...}

 

expression

WeakMutableSet.by(key_comp)

Like MutableSet, but creates a set where an element is removed from the set by a garbage collection when the element is reachable only by enumerating the set’s elements.

value

def Set.empty :: Set = Set{}

 

binding operator

Set.empty

 

value

def ReadableSet.empty :: ReadableSet = Set{}

 

binding operator

ReadableSet.empty

An empty set. The Set.empty binding form differs from from Set{}, because Set.empty matches only an empty immutable set, while Set{} matches any immutable set.

The ReadableSet.empty binding form matches an empty set whether it is mutable or immutable.

Corresponding to the binding forms, Set.empty and ReadableSet.empty are bound to Set{} with appropriate static information.

> Set.empty

Set{}

> match Set()

  | Set.empty: "empty set"

  | _: #false

"empty set"

> match MutableSet()

  | Set.empty: "empty immutable set"

  | _: #false

#false

> match MutableSet()

  | ReadableSet.empty: "empty set for now"

  | _: #false

"empty set for now"

Returns the number of values in st.

> Set.length({"a", "b"})

2

> Set.length(Set{})

0

> {"a", "b"}.length()

2

> Set{}.length()

0

method

method Set.get(st :: ReadableSet, val :: Any) :: Boolean

 

method

method Set.has_element(st :: ReadableSet, val :: Any) :: Boolean

Equivalent to st[val] (with the default implicit #%index form). Returns #true if val is in st, #false otherwise.

> {"a", "b"}.get("a")

#true

> {"a", "b"}.has_element("a")

#true

> {"a", "b"}["a"]

#true

method

method (st :: Set).append(another_st :: Set, ...) :: Set

 

method

method (st :: Set).union(another_st :: Set, ...) :: Set

Functionally appends st and another_sts, like the ++ operator (but without the special optimization for adding a single element).

Even when another_st uses a different map configuration than st, the map configuration of st is preserved in the result set. Conceptually, in the binary case, each element from the right set is added to the left set.

> {1, 2, 3}.append({2, 3}, {3, 4})

{1, 2, 3, 4}

> {1, 2, 3}.union({2, 3}, {3, 4})

{1, 2, 3, 4}

> {1, 2, 3} ++ {2, 3} ++ {3, 4}

{1, 2, 3, 4}

> {1, 2, 3}.append(

    Set.by(is_same_number_or_object){2, 3},

    Set.by(is_now){3, 4},

  )

{1, 2, 3, 4}

> {1, 2, 3}.union(

    Set.by(is_same_number_or_object){2, 3},

    Set.by(is_now){3, 4},

  )

{1, 2, 3, 4}

> {1, 2, 3}

    ++ Set.by(is_same_number_or_object){2, 3}

    ++ Set.by(is_now){3, 4}

{1, 2, 3, 4}

method

method (st :: Set).intersect(another_st :: Set, ...) :: Set

Returns the intersection of st and another_sts.

Even when some another_st uses a different map configuration than st, the map configuration of st is preserved in the result set.

> {1, 2, 3}.intersect({2, 3}, {3, 4})

{3}

> {1, 2, 3}.intersect(

    Set.by(is_same_number_or_object){2, 3},

    Set.by(is_now){3, 4},

  )

{3}

method

method (st :: Set).remove(val :: Any) :: Set

Returns a set like st but without val, if it is present.

> {1, 2, 3}.remove(2)

{1, 3}

> {1, 2, 3}.remove(4)

{1, 2, 3}

method

method (st :: MutableSet).set(val :: Any, in :: Any)

  :: Void

Equivalent to st[val] := in (with the default implicit #%index form). Changes st to remove val if in is #false, otherwise add val.

> def s = MutableSet{1, 2, 3}

> s.set(1, #false)

> s

MutableSet{2, 3}

> s[1] := #true

> s

MutableSet{1, 2, 3}

method

method (st :: MutableSet).delete(val :: Any)

  :: Void

Changes st to remove val, if it is present.

> def s = MutableSet{1, 2, 3}

> s.delete(2)

> s

MutableSet{1, 3}

method

method Set.to_list(st :: ReadableSet,

                   try_sort :: Any = #false)

  :: List

Returns the elements of st in a list. If try_sort is true, then the elements are sorted to the degree that a built-in comparison can sort them. Note that sets do not implement Listable, because the order of elements is unspecified.

> {1, 2, 3}.to_list(#true)

[1, 2, 3]

Creates a mutable set whose initial content matches st.

> def s = {"a", "b"}

> s.copy()

MutableSet{"a", "b"}

> def s = MutableSet{"a", "b"}

> s.copy()

MutableSet{"a", "b"}

Returns an immutable set whose content matches st. If st is immutable, then it is the result.

> def s = {"a", "b"}

> s.snapshot()

{"a", "b"}

> s.snapshot() === s

#true

> def s = MutableSet{"a", "b"}

> s.snapshot()

{"a", "b"}

Implements Sequenceable by returning a sequence of st’s elements in an unspecified order.