2024-08-18 15:23:32 -07:00
|
|
|
# Tables
|
|
|
|
|
|
|
|
Tables are Tomo's associative mapping structure, also known as a Dictionary or
|
|
|
|
Map. Tables are efficiently implemented as a hash table that preserves
|
|
|
|
insertion order and has fast access to keys and values as array slices. Tables
|
|
|
|
support *all* types as both keys and values.
|
|
|
|
|
2024-08-18 17:00:21 -07:00
|
|
|
## Syntax
|
|
|
|
|
2025-04-02 13:14:20 -07:00
|
|
|
Tables are written using `{}` curly braces with `=` equals signs associating key
|
2024-08-18 17:00:21 -07:00
|
|
|
expressions with value expressions and commas between entries:
|
|
|
|
|
|
|
|
```tomo
|
2025-04-02 13:14:20 -07:00
|
|
|
table := {"A"=10, "B"=20}
|
2024-08-18 17:00:21 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
Empty tables must specify the key and value types explicitly:
|
|
|
|
|
|
|
|
```tomo
|
2025-04-02 13:14:20 -07:00
|
|
|
empty := {:Text=Int}
|
2024-08-18 17:00:21 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
For type annotations, a table that maps keys with type `K` to values of type
|
2025-04-02 13:14:20 -07:00
|
|
|
`V` is written as `{K=V}`.
|
2024-08-18 17:00:21 -07:00
|
|
|
|
|
|
|
### Comprehensions
|
|
|
|
|
|
|
|
Similar to arrays, tables can use comprehensions to dynamically construct tables:
|
|
|
|
|
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
t := {i=10*i for i in 10}
|
|
|
|
t := {i=10*i for i in 10 if i mod 2 == 0}
|
|
|
|
t := {-1=-10, i=10*i for i in 10}
|
2024-08-18 17:00:21 -07:00
|
|
|
```
|
|
|
|
|
2024-11-30 12:50:54 -08:00
|
|
|
## Accessing Values
|
2024-09-11 21:55:43 -07:00
|
|
|
|
2024-11-30 12:50:54 -08:00
|
|
|
Table values can be accessed with square bracket indexing. The result is an
|
|
|
|
optional value:
|
2024-09-11 21:55:43 -07:00
|
|
|
|
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
table := {"A"=1, "B"=2}
|
2024-11-30 12:50:54 -08:00
|
|
|
>> table["A"]
|
2025-04-02 13:14:20 -07:00
|
|
|
= 1?
|
2024-11-30 12:50:54 -08:00
|
|
|
>> table["missing"]
|
2025-04-02 13:14:20 -07:00
|
|
|
= none:Int
|
2024-11-30 12:50:54 -08:00
|
|
|
```
|
|
|
|
|
|
|
|
As with all optional values, you can use the `!` postfix operator to assert
|
2024-12-07 13:04:25 -08:00
|
|
|
that the value is non-none (and create a runtime error if it is), or you can
|
|
|
|
use the `or` operator to provide a fallback value in the case that it's none:
|
2024-11-30 12:50:54 -08:00
|
|
|
|
|
|
|
```tomo
|
|
|
|
>> table["A"]!
|
2025-04-02 13:14:20 -07:00
|
|
|
= 1
|
2024-11-30 12:50:54 -08:00
|
|
|
|
|
|
|
>> table["missing"] or -1
|
2025-04-02 13:14:20 -07:00
|
|
|
= -1
|
2024-09-11 21:55:43 -07:00
|
|
|
```
|
|
|
|
|
2024-08-18 17:00:21 -07:00
|
|
|
### Fallback Tables
|
|
|
|
|
|
|
|
Tables can specify a fallback table that is used when looking up a value if it
|
|
|
|
is not found in the table itself:
|
|
|
|
|
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
t := {"A"=10}
|
|
|
|
t2 := {"B"=20; fallback=t}
|
2024-11-30 12:50:54 -08:00
|
|
|
>> t2["A"]
|
2025-04-02 13:14:20 -07:00
|
|
|
= 10?
|
2024-08-18 17:00:21 -07:00
|
|
|
```
|
|
|
|
|
2024-08-18 17:28:16 -07:00
|
|
|
The fallback is available by the `.fallback` field, which returns an optional
|
2024-11-30 12:50:54 -08:00
|
|
|
table value:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
>> t2.fallback
|
2025-04-02 13:14:20 -07:00
|
|
|
= {"A"=10}?
|
2024-11-30 12:50:54 -08:00
|
|
|
>> t.fallback
|
2025-04-02 13:14:20 -07:00
|
|
|
= none:{Text=Int}
|
2024-11-30 12:50:54 -08:00
|
|
|
```
|
|
|
|
|
2025-04-02 13:14:20 -07:00
|
|
|
### Default Values
|
|
|
|
|
|
|
|
Tables can specify a default value which will be returned if a value is not
|
|
|
|
present in the table or its fallback (if any).
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
counts := &{"foo"=12; default=0}
|
|
|
|
>> counts["foo"]
|
|
|
|
= 12
|
|
|
|
>> counts["baz"]
|
|
|
|
= 0
|
|
|
|
counts["baz"] += 1
|
|
|
|
>> counts["baz"]
|
|
|
|
= 1
|
|
|
|
```
|
|
|
|
|
|
|
|
When values are accessed from a table with a default value, the return type
|
|
|
|
is non-optional (because a value will always be present).
|
|
|
|
|
2024-11-30 12:50:54 -08:00
|
|
|
## Setting Values
|
|
|
|
|
|
|
|
You can assign a new key/value mapping or overwrite an existing one using
|
|
|
|
`:set(key, value)` or an `=` assignment statement:
|
|
|
|
|
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
t := {"A"=1, "B"=2}
|
2024-11-30 12:50:54 -08:00
|
|
|
t["B"] = 222
|
|
|
|
t["C"] = 333
|
|
|
|
>> t
|
2025-01-12 13:49:58 -08:00
|
|
|
= {"A"=1, "B"=222, "C"=333}
|
2024-11-30 12:50:54 -08:00
|
|
|
```
|
2024-08-18 17:28:16 -07:00
|
|
|
|
|
|
|
## Length
|
|
|
|
|
|
|
|
Table length can be accessed by the `.length` field:
|
|
|
|
|
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
>> {"A"=10, "B"=20}.length
|
2024-08-18 17:28:16 -07:00
|
|
|
= 2
|
|
|
|
```
|
|
|
|
|
2024-08-18 17:00:21 -07:00
|
|
|
## Accessing Keys and Values
|
|
|
|
|
|
|
|
The keys and values of a table can be efficiently accessed as arrays using a
|
|
|
|
constant-time immutable slice of the internal data from the table:
|
|
|
|
|
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
t := {"A"=10, "B"=20}
|
2024-08-18 17:00:21 -07:00
|
|
|
>> t.keys
|
|
|
|
= ["A", "B"]
|
|
|
|
>> t.values
|
|
|
|
= [10, 20]
|
|
|
|
```
|
|
|
|
|
|
|
|
## Iteration
|
|
|
|
|
|
|
|
You can iterate over the key/value pairs in a table like this:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
for key, value in table:
|
|
|
|
...
|
|
|
|
|
|
|
|
for key in table:
|
|
|
|
...
|
|
|
|
```
|
|
|
|
|
|
|
|
Table iteration operates over the value of the table when the loop began, so
|
|
|
|
modifying the table during iteration is safe and will not result in the loop
|
|
|
|
iterating over any of the new values.
|
2024-08-18 15:23:32 -07:00
|
|
|
|
|
|
|
## Table Methods
|
|
|
|
|
2025-04-02 13:14:20 -07:00
|
|
|
- [`func bump(t:&{K=V}, key: K, amount: Int = 1 -> Void)`](#bump)
|
|
|
|
- [`func clear(t:&{K=V})`](#clear)
|
|
|
|
- [`func get(t:{K=V}, key: K -> V?)`](#get)
|
|
|
|
- [`func has(t:{K=V}, key: K -> Bool)`](#has)
|
|
|
|
- [`func remove(t:{K=V}, key: K -> Void)`](#remove)
|
|
|
|
- [`func set(t:{K=V}, key: K, value: V -> Void)`](#set)
|
2025-03-04 21:21:30 -08:00
|
|
|
|
2024-08-18 15:23:32 -07:00
|
|
|
### `bump`
|
|
|
|
Increments the value associated with a key by a specified amount. If the key is
|
|
|
|
not already in the table, its value will be assumed to be zero.
|
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-04-02 13:14:20 -07:00
|
|
|
func bump(t:&{K=V}, key: K, amount: Int = 1 -> Void)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
- `t`: The reference to the table.
|
2024-08-18 15:23:32 -07:00
|
|
|
- `key`: The key whose value is to be incremented.
|
|
|
|
- `amount`: The amount to increment the value by (default: 1).
|
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
Nothing.
|
|
|
|
|
|
|
|
**Example:**
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
>> t := {"A"=1}
|
2024-08-18 15:23:32 -07:00
|
|
|
t:bump("A")
|
|
|
|
t:bump("B", 10)
|
|
|
|
>> t
|
2025-01-12 13:49:58 -08:00
|
|
|
= {"A"=2, "B"=10}
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `clear`
|
|
|
|
Removes all key-value pairs from the table.
|
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-04-02 13:14:20 -07:00
|
|
|
func clear(t:&{K=V})
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
- `t`: The reference to the table.
|
2024-08-18 15:23:32 -07:00
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
Nothing.
|
|
|
|
|
|
|
|
**Example:**
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2024-08-18 15:23:32 -07:00
|
|
|
>> t:clear()
|
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `get`
|
2025-04-02 13:14:20 -07:00
|
|
|
Retrieves the value associated with a key, or returns `none` if the key is not present.
|
|
|
|
**Note:** default values for the table are ignored.
|
2024-08-18 15:23:32 -07:00
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-04-02 13:14:20 -07:00
|
|
|
func get(t:{K=V}, key: K -> V?)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
- `t`: The table.
|
|
|
|
- `key`: The key whose associated value is to be retrieved.
|
|
|
|
|
|
|
|
**Returns:**
|
2025-04-02 13:14:20 -07:00
|
|
|
The value associated with the key or `none` if the key is not found.
|
2024-08-18 15:23:32 -07:00
|
|
|
|
|
|
|
**Example:**
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
>> t := {"A"=1, "B"=2}
|
2024-08-18 15:23:32 -07:00
|
|
|
>> t:get("A")
|
2025-04-02 13:14:20 -07:00
|
|
|
= 1?
|
2024-08-18 15:23:32 -07:00
|
|
|
|
2024-09-11 21:55:43 -07:00
|
|
|
>> t:get("????")
|
2025-04-02 13:14:20 -07:00
|
|
|
= none:Int
|
2024-08-18 15:23:32 -07:00
|
|
|
|
2024-09-11 21:55:43 -07:00
|
|
|
>> t:get("A")!
|
2025-04-02 13:14:20 -07:00
|
|
|
= 1
|
2024-08-18 15:23:32 -07:00
|
|
|
|
2024-09-16 13:06:19 -07:00
|
|
|
>> t:get("????") or 0
|
2025-04-02 13:14:20 -07:00
|
|
|
= 0
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `has`
|
|
|
|
Checks if the table contains a specified key.
|
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-04-02 13:14:20 -07:00
|
|
|
func has(t:{K=V}, key: K -> Bool)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
- `t`: The table.
|
|
|
|
- `key`: The key to check for presence.
|
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
`yes` if the key is present, `no` otherwise.
|
|
|
|
|
|
|
|
**Example:**
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
>> {"A"=1, "B"=2}:has("A")
|
2024-08-18 15:23:32 -07:00
|
|
|
= yes
|
2025-01-12 13:49:58 -08:00
|
|
|
>> {"A"=1, "B"=2}:has("xxx")
|
2024-08-18 15:23:32 -07:00
|
|
|
= no
|
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `remove`
|
|
|
|
Removes the key-value pair associated with a specified key.
|
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-04-02 13:14:20 -07:00
|
|
|
func remove(t:{K=V}, key: K -> Void)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
- `t`: The reference to the table.
|
2024-08-18 15:23:32 -07:00
|
|
|
- `key`: The key of the key-value pair to remove.
|
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
Nothing.
|
|
|
|
|
|
|
|
**Example:**
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
t := {"A"=1, "B"=2}
|
2024-08-18 15:23:32 -07:00
|
|
|
t:remove("A")
|
|
|
|
>> t
|
2025-01-12 13:49:58 -08:00
|
|
|
= {"B"=2}
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `set`
|
|
|
|
Sets or updates the value associated with a specified key.
|
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-04-02 13:14:20 -07:00
|
|
|
func set(t:{K=V}, key: K, value: V -> Void)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
2024-10-09 10:48:45 -07:00
|
|
|
- `t`: The reference to the table.
|
2024-08-18 15:23:32 -07:00
|
|
|
- `key`: The key to set or update.
|
|
|
|
- `value`: The value to associate with the key.
|
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
Nothing.
|
|
|
|
|
|
|
|
**Example:**
|
2024-10-09 10:48:45 -07:00
|
|
|
```tomo
|
2025-01-12 13:49:58 -08:00
|
|
|
t := {"A"=1, "B"=2}
|
2024-08-18 15:23:32 -07:00
|
|
|
t:set("C", 3)
|
|
|
|
>> t
|
2025-01-12 13:49:58 -08:00
|
|
|
= {"A"=1, "B"=2, "C"=3}
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|