2024-09-11 12:12:00 -07:00
|
|
|
# Optional Values
|
|
|
|
|
|
|
|
A very common use case is values that may or may not be present. You could
|
|
|
|
represent this case using enums like so:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
enum MaybeInt(AnInt(x:Int), NoInt)
|
|
|
|
|
|
|
|
func maybe_takes_int(maybe_x:MaybeInt):
|
|
|
|
when maybe_x is AnInt(x):
|
|
|
|
say("Got an int: $x")
|
|
|
|
else:
|
|
|
|
say("Got nothing")
|
|
|
|
```
|
|
|
|
|
|
|
|
However, it's overly onerous to have to define a separate type for each
|
|
|
|
situation where you might want to not have a value. Instead, Tomo has
|
|
|
|
built-in support for optional types:
|
|
|
|
|
|
|
|
```
|
|
|
|
func maybe_takes_int(x:Int?):
|
|
|
|
if x:
|
|
|
|
say("Got an int: $x")
|
|
|
|
else:
|
|
|
|
say("Got nothing")
|
|
|
|
```
|
|
|
|
|
|
|
|
This establishes a common language for talking about optional values without
|
|
|
|
having to use a more generalized form of `enum` which may have different naming
|
2024-11-21 10:00:53 -08:00
|
|
|
conventions and which would generate a lot of unnecessary code.
|
|
|
|
|
|
|
|
## Syntax
|
|
|
|
|
|
|
|
Optional types are written using a `?` after the type name. So, an optional
|
|
|
|
integer would be written as `Int?` and an optional array of texts would be
|
|
|
|
written as `[Text]?`.
|
|
|
|
|
2024-12-07 13:04:25 -08:00
|
|
|
None can be written explicitly using `none` with a type annotation. For
|
2024-11-24 13:36:27 -08:00
|
|
|
example, if you wanted to declare a variable that could be either an integer
|
2024-12-07 13:04:25 -08:00
|
|
|
value or `none` and initialize it as none, you would write it as:
|
2024-11-21 10:00:53 -08:00
|
|
|
|
|
|
|
```tomo
|
2024-12-07 13:04:25 -08:00
|
|
|
x := none:Int
|
2024-11-21 10:00:53 -08:00
|
|
|
```
|
|
|
|
|
|
|
|
Similarly, if you wanted to declare a variable that could be an array of texts
|
2024-11-24 13:36:27 -08:00
|
|
|
or none and initialize it as none, you would write:
|
2024-11-21 10:00:53 -08:00
|
|
|
|
|
|
|
```tomo
|
|
|
|
x := ![Text]
|
|
|
|
```
|
|
|
|
|
2024-11-24 13:36:27 -08:00
|
|
|
If you want to declare a variable and initialize it with a non-none value, but
|
2024-12-07 13:04:25 -08:00
|
|
|
keep open the possibility of assigning `none` later, you can use the postfix
|
2024-11-24 13:36:27 -08:00
|
|
|
`?` operator to indicate that a value is optional:
|
2024-11-21 10:00:53 -08:00
|
|
|
|
|
|
|
```tomo
|
|
|
|
x := 5?
|
2024-11-24 13:36:27 -08:00
|
|
|
# Later on, assign none:
|
2024-11-21 10:00:53 -08:00
|
|
|
x = !Int
|
|
|
|
```
|
|
|
|
|
|
|
|
## Type Inference
|
|
|
|
|
2024-12-07 13:04:25 -08:00
|
|
|
For convenience, `none` can also be written without the explicit type
|
2024-11-24 13:36:27 -08:00
|
|
|
annotation for any type in situations where the compiler knows what type of
|
|
|
|
optional value is expected:
|
2024-11-21 10:00:53 -08:00
|
|
|
|
|
|
|
- When assigning to a variable that has already been declared as optional.
|
|
|
|
- When returning from a function with an explicit optional return type.
|
|
|
|
- When passing an argument to a function with an optional argument type.
|
|
|
|
|
|
|
|
Here are some examples:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
x := 5?
|
2024-12-07 13:04:25 -08:00
|
|
|
x = none
|
2024-11-21 10:00:53 -08:00
|
|
|
|
|
|
|
func doop(arg:Int?)->Text?:
|
2024-12-07 13:04:25 -08:00
|
|
|
return none
|
2024-11-21 10:00:53 -08:00
|
|
|
|
2024-12-07 13:04:25 -08:00
|
|
|
doop(none)
|
2024-11-21 10:00:53 -08:00
|
|
|
```
|
|
|
|
|
2024-11-24 13:36:27 -08:00
|
|
|
Non-none values can also be automatically promoted to optional values without
|
2024-11-21 10:00:53 -08:00
|
|
|
the need for an explicit `?` operator in the cases listed above:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
x := !Int
|
|
|
|
x = 5
|
|
|
|
|
|
|
|
func doop(arg:Int?)->Text?:
|
|
|
|
return "okay"
|
|
|
|
|
|
|
|
doop(123)
|
|
|
|
```
|
|
|
|
|
2024-11-24 13:36:27 -08:00
|
|
|
## None Checking
|
2024-09-11 20:17:03 -07:00
|
|
|
|
2024-12-07 13:04:25 -08:00
|
|
|
In addition to using conditionals to check for `none`, you can also use `or` to
|
2024-11-24 13:36:27 -08:00
|
|
|
get a non-none value by either providing an alternative non-none value or by
|
|
|
|
providing an early out statement like `return`/`skip`/`stop` or a function with
|
|
|
|
an `Abort` type like `fail()` or `exit()`:
|
2024-09-11 20:17:03 -07:00
|
|
|
|
|
|
|
```tomo
|
|
|
|
maybe_x := 5?
|
2024-09-16 13:06:19 -07:00
|
|
|
>> maybe_x or -1
|
2024-09-11 20:17:03 -07:00
|
|
|
= 5 : Int
|
2024-09-16 13:06:19 -07:00
|
|
|
>> maybe_x or fail("No value!")
|
2024-09-11 20:17:03 -07:00
|
|
|
= 5 : Int
|
|
|
|
|
|
|
|
maybe_x = !Int
|
2024-09-16 13:06:19 -07:00
|
|
|
>> maybe_x or -1
|
2024-09-11 20:17:03 -07:00
|
|
|
= -1 : Int
|
2024-09-16 13:06:19 -07:00
|
|
|
>> maybe_x or fail("No value!")
|
2024-09-11 20:17:03 -07:00
|
|
|
# Failure!
|
2024-09-15 13:42:42 -07:00
|
|
|
|
2024-09-16 13:06:19 -07:00
|
|
|
func do_stuff(matches:[Text]):
|
|
|
|
pass
|
2024-09-15 13:42:42 -07:00
|
|
|
|
2024-09-16 13:06:19 -07:00
|
|
|
for line in lines:
|
|
|
|
matches := line:matches($/{..},{..}/) or skip
|
2024-11-24 13:36:27 -08:00
|
|
|
# The `or skip` above means that if we're here, `matches` is non-none:
|
2024-09-16 13:06:19 -07:00
|
|
|
do_stuff(matches)
|
2024-09-11 20:17:03 -07:00
|
|
|
```
|
2024-11-21 10:00:53 -08:00
|
|
|
|
|
|
|
## Implementation Notes
|
|
|
|
|
|
|
|
The implementation of optional types is highly efficient and has no memory
|
2025-01-02 13:24:07 -08:00
|
|
|
overhead for pointers, collection types (arrays, sets, tables), booleans,
|
|
|
|
texts, enums, nums, or integers (`Int` type only). This is done by using
|
|
|
|
carefully chosen values, such as `0` for pointers, `2` for booleans, or a
|
2024-11-21 10:00:53 -08:00
|
|
|
negative length for arrays. However, for fixed-size integers (`Int64`, `Int32`,
|
|
|
|
`Int16`, and `Int8`), bytes, and structs, an additional byte is required for
|
2024-11-24 13:36:27 -08:00
|
|
|
out-of-band information about whether the value is none or not.
|
2024-11-21 10:00:53 -08:00
|
|
|
|
2024-11-24 13:36:27 -08:00
|
|
|
Floating point numbers (`Num` and `Num32`) use `NaN` to represent none, so
|
|
|
|
optional nums should be careful to avoid using `NaN` as a non-none value. This
|
2024-11-21 10:00:53 -08:00
|
|
|
option was chosen to minimize the memory overhead of optional nums and because
|
|
|
|
`NaN` literally means "not a number".
|