9.0.0.2

2.6 Conditionals and Pattern-Matching Dispatch🔗ℹ

The && and || operators are short-circuiting “and” and ”or” forms. The || operator returns the first non-#false value, and && returns the last non-#false value.

> 1 < 2 && "ok"

"ok"

Comparison operators and ! (for “not”) have higher precedence than && and ||, while && has higher precedence than ||. Arithmetic operators have higher precedence than comparison operators, ||, &&, but they have no precedence relative to !. The == operator reports structural equality independent of mutations, while the === operator reports object equality. Comparison operators are non-associative and have no precedence relationship with each other.

The if form expects a test expression followed by alternatives with two |s. The first | holds the “then” branch, and the second | holds the “else” branch:

> if 1 == 2

  | "same"

  | "different"

"different"

Although an if could be nested further in the “else” branch to implement an “if” ... “else if” ... “else if” ... combination, the cond form supports that combination better. It expects alternatives where each | has a test expression followed by a block. Evaluating the cond form dispatches to the block after first test that produces a non-#false value. The ~else keyword can be used in place of a last test.

fun fib(n):

  cond

  | n == 0: 1

  | n == 1: 1

  | ~else: fib(n-1) + fib(n-2)

> fib(5)

8

If there’s no ~else case and no matching case, then cond reports an error at run time.

Although cond is better than if for fib, the match form is even better. The match form expects an expression and then alternatives where each | has a binding pattern followed by a block. The match form evaluates that first expression, and then it dispatches to the first block whose pattern accepts the expression’s value. Similar to cond, match supports ~else in place of a final binding pattern, but using the binding operator _ also works.

fun fib(n):

  match n

  | 0: 1

  | 1: 1

  | _: fib(n-1) + fib(n-2)

This kind of immediate pattern-matching dispatch on a function argument is common enough that fun supports it directly, fusing the function declaration and the pattern match, like this:

fun

| fib(0): 1

| fib(1): 1

| fib(n): fib(n-1) + fib(n-2)

There’s no ~else for this fused form, but _ can be useful in catch-call clauses where the argument is not used. Also, the function name and all relevant argument positions have to be repeated in every case, but that’s often a readable trade-off. Match-dispatching functions cannot have optional arguments, but different cases can have different numbers of arguments, and a call will find a matching case with the right number of arguments.

fun

| hello(name):

    "Hello, " +& name    // +& coerces to strings and concatenates

| hello(first, last):

    hello(first +& " " +& last)

> hello("World")

"Hello, World"

> hello("Inigo", "Montoya")

"Hello, Inigo Montoya"

To write a result annotation just once in a function definition with multiple cases, use the function name after fun, then :: or :~, the annotation, and then the cases:

fun fib :: PosInt:

| fib(0): 1

| fib(1): 1

| fib(n :: NonnegInt): fib(n-1) + fib(n-2)

The if, cond, and match forms are best for writing a single conditional expression where each branch has similar importance. However, it is often the case that some condition needs to be gotten out of the way before the “real logic” of an operation can happen. Consider this code:

fun show_user_stats(user_id):

  cond

  | user_id == "": #false

  | ~else:

      let user = get_user(user_id)

      let document_count = user.get_documents().length()

      let message = @str{User @user."elem" has @document_count documents.}

      println(message)

> show_user_stats("bf4afcb")

User Alice has 3 documents.

> show_user_stats("")

#false

In these instances, the guard form can be used. A guard expression checks that a condition is true, and if it isn’t, short-circuits the surrounding block with a given alternative. The above code can be rewritten using guard expressions like so:

fun show_user_stats(user_id):

  guard user_id != "" | #false

  let user = get_user(user_id)

  let document_count = user.get_documents().length()

  let message = @str{User @user."elem" has @document_count documents.}

  println(message)

> show_user_stats("bf4afcb")

User Alice has 3 documents.

> show_user_stats("")

#false

A pattern matching variant, guard.let, is also available. The guard.let form checks that a value matches a pattern, otherwise it short-circuits with a given alternative just like guard. Here we can use it to check that two lists are equal:

fun lists_equal(xs, ys):

  guard.let [x, & rest_xs] = xs | ys == []

  guard.let [y, & rest_ys] = ys | #false

  x == y && lists_equal(rest_xs, rest_ys)

> lists_equal([1, 2, 3], [1, 2, 3])

#true

> lists_equal([1], [1, 2, 3])

#false

> lists_equal([1, 2, 3], [1])

#false

> lists_equal([1, 2, 3], [1, 2, 10000])

#false

A guard form only escapes from its immediately enclosing block. Non-local escapes can be implemented with try, either by throwing and catching an exception or calling an escape function bound by try.