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.
|
|
|
|
|
|
|
|
Tables do not support square bracket indexing (`t[key]`), but instead rely on
|
|
|
|
the methods `:get(key)` and `:set(key, value)`. This is explicit to avoid
|
|
|
|
hiding the fact that table lookups and table insertion are performing function
|
|
|
|
calls and have edge conditions like a failure to find an entry.
|
|
|
|
|
2024-08-18 17:00:21 -07:00
|
|
|
## Syntax
|
|
|
|
|
|
|
|
Tables are written using `{}` curly braces with `:` colons associating key
|
|
|
|
expressions with value expressions and commas between entries:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
table := {"A": 10, "B": 20}
|
|
|
|
```
|
|
|
|
|
|
|
|
Empty tables must specify the key and value types explicitly:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
empty := {:Text:Int}
|
|
|
|
```
|
|
|
|
|
|
|
|
For type annotations, a table that maps keys with type `K` to values of type
|
|
|
|
`V` is written as `{K:V}`.
|
|
|
|
|
|
|
|
### Comprehensions
|
|
|
|
|
|
|
|
Similar to arrays, tables can use comprehensions to dynamically construct tables:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
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-09-11 21:55:43 -07:00
|
|
|
### Getting Values
|
|
|
|
|
|
|
|
To get a value from a table, use `:get(key)`, which returns an _optional_
|
|
|
|
value, depending on whether it was present in the table or not. For convenience,
|
|
|
|
you can use the `!` postifx operator to perform a check to ensure that the value
|
|
|
|
was found or error if it wasn't:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
>> t := {"x":1, "y":2}
|
|
|
|
>> t:get("x")
|
|
|
|
= 1?
|
|
|
|
>> t:get("????")
|
|
|
|
= !Int
|
|
|
|
>> t:get("x")!
|
|
|
|
= 1
|
|
|
|
```
|
|
|
|
|
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
|
|
|
|
t := {"A": 10}
|
|
|
|
t2 := {"B": 20; fallback=t}
|
|
|
|
>> t2:get("A")
|
|
|
|
= 10
|
|
|
|
```
|
|
|
|
|
2024-08-18 17:28:16 -07:00
|
|
|
The fallback is available by the `.fallback` field, which returns an optional
|
2024-09-11 12:01:51 -07:00
|
|
|
table value.
|
2024-08-18 17:28:16 -07:00
|
|
|
|
|
|
|
## Length
|
|
|
|
|
|
|
|
Table length can be accessed by the `.length` field:
|
|
|
|
|
|
|
|
```tomo
|
|
|
|
>> {"A":10, "B":20}.length
|
|
|
|
= 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
|
|
|
|
t := {"A": 10, "B": 20}
|
|
|
|
>> 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
|
|
|
|
|
|
|
|
### `bump`
|
|
|
|
|
|
|
|
**Description:**
|
|
|
|
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.
|
|
|
|
|
|
|
|
**Usage:**
|
|
|
|
```markdown
|
2024-10-09 10:26:28 -07:00
|
|
|
bump(t:{K:V}, key: K, amount: Int = 1 -> Void)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
**Parameters:**
|
|
|
|
|
|
|
|
- `t`: The mutable reference to the table.
|
|
|
|
- `key`: The key whose value is to be incremented.
|
|
|
|
- `amount`: The amount to increment the value by (default: 1).
|
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
Nothing.
|
|
|
|
|
|
|
|
**Example:**
|
|
|
|
```markdown
|
|
|
|
>> t := {"A":1}
|
|
|
|
t:bump("A")
|
|
|
|
t:bump("B", 10)
|
|
|
|
>> t
|
|
|
|
= {"A": 2, "B": 10}
|
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `clear`
|
|
|
|
|
|
|
|
**Description:**
|
|
|
|
Removes all key-value pairs from the table.
|
|
|
|
|
|
|
|
**Usage:**
|
|
|
|
```markdown
|
2024-10-09 10:26:28 -07:00
|
|
|
t:clear()
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
**Parameters:**
|
|
|
|
|
|
|
|
- `t`: The mutable reference to the table.
|
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
Nothing.
|
|
|
|
|
|
|
|
**Example:**
|
|
|
|
```markdown
|
|
|
|
>> t:clear()
|
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `get`
|
|
|
|
|
|
|
|
**Description:**
|
2024-09-11 21:55:43 -07:00
|
|
|
Retrieves the value associated with a key, or returns null if the key is not present.
|
2024-08-18 15:23:32 -07:00
|
|
|
|
|
|
|
**Usage:**
|
|
|
|
```markdown
|
2024-10-09 10:26:28 -07:00
|
|
|
t:get(key: K -> V?)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
**Parameters:**
|
|
|
|
|
|
|
|
- `t`: The table.
|
|
|
|
- `key`: The key whose associated value is to be retrieved.
|
|
|
|
|
|
|
|
**Returns:**
|
2024-09-11 21:55:43 -07:00
|
|
|
The value associated with the key or null if the key is not found.
|
2024-08-18 15:23:32 -07:00
|
|
|
|
|
|
|
**Example:**
|
|
|
|
```markdown
|
|
|
|
>> t := {"A":1, "B":2}
|
|
|
|
>> t:get("A")
|
2024-09-11 21:55:43 -07:00
|
|
|
= 1? : Int?
|
2024-08-18 15:23:32 -07:00
|
|
|
|
2024-09-11 21:55:43 -07:00
|
|
|
>> t:get("????")
|
|
|
|
= !Int : Int?
|
2024-08-18 15:23:32 -07:00
|
|
|
|
2024-09-11 21:55:43 -07:00
|
|
|
>> t:get("A")!
|
|
|
|
= 1 : Int
|
2024-08-18 15:23:32 -07:00
|
|
|
|
2024-09-16 13:06:19 -07:00
|
|
|
>> t:get("????") or 0
|
2024-09-11 21:55:43 -07:00
|
|
|
= 0 : Int
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `has`
|
|
|
|
|
|
|
|
**Description:**
|
|
|
|
Checks if the table contains a specified key.
|
|
|
|
|
|
|
|
**Usage:**
|
|
|
|
```markdown
|
2024-10-09 10:26:28 -07:00
|
|
|
has(t:{K:V}, key: K -> Bool)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
**Parameters:**
|
|
|
|
|
|
|
|
- `t`: The table.
|
|
|
|
- `key`: The key to check for presence.
|
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
`yes` if the key is present, `no` otherwise.
|
|
|
|
|
|
|
|
**Example:**
|
|
|
|
```markdown
|
|
|
|
>> {"A":1, "B":2}:has("A")
|
|
|
|
= yes
|
|
|
|
>> {"A":1, "B":2}:has("xxx")
|
|
|
|
= no
|
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `remove`
|
|
|
|
|
|
|
|
**Description:**
|
|
|
|
Removes the key-value pair associated with a specified key.
|
|
|
|
|
|
|
|
**Usage:**
|
|
|
|
```markdown
|
2024-10-09 10:26:28 -07:00
|
|
|
remove(t:{K:V}, key: K -> Void)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
**Parameters:**
|
|
|
|
|
|
|
|
- `t`: The mutable reference to the table.
|
|
|
|
- `key`: The key of the key-value pair to remove.
|
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
Nothing.
|
|
|
|
|
|
|
|
**Example:**
|
|
|
|
```markdown
|
|
|
|
t := {"A":1, "B":2}
|
|
|
|
t:remove("A")
|
|
|
|
>> t
|
|
|
|
= {"B": 2}
|
|
|
|
```
|
|
|
|
|
|
|
|
---
|
|
|
|
|
|
|
|
### `set`
|
|
|
|
|
|
|
|
**Description:**
|
|
|
|
Sets or updates the value associated with a specified key.
|
|
|
|
|
|
|
|
**Usage:**
|
|
|
|
```markdown
|
2024-10-09 10:26:28 -07:00
|
|
|
set(t:{K:V}, key: K, value: V -> Void)
|
2024-08-18 15:23:32 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
**Parameters:**
|
|
|
|
|
|
|
|
- `t`: The mutable reference to the table.
|
|
|
|
- `key`: The key to set or update.
|
|
|
|
- `value`: The value to associate with the key.
|
|
|
|
|
|
|
|
**Returns:**
|
|
|
|
Nothing.
|
|
|
|
|
|
|
|
**Example:**
|
|
|
|
```markdown
|
|
|
|
t := {"A": 1, "B": 2}
|
|
|
|
t:set("C", 3)
|
|
|
|
>> t
|
|
|
|
= {"A": 1, "B": 2, "C": 3}
|
|
|
|
```
|