mirror of
https://github.com/luau-lang/rfcs.git
synced 2025-04-05 11:00:58 +01:00
Create monomorphic-type-bindings-for-generics.md
This commit is contained in:
parent
12a39c3fdb
commit
213ec9f8d4
1 changed files with 56 additions and 0 deletions
56
docs/monomorphic-type-bindings-for-generics.md
Normal file
56
docs/monomorphic-type-bindings-for-generics.md
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
# Monomorphic Type Bindings for Generics
|
||||||
|
|
||||||
|
## Summary
|
||||||
|
|
||||||
|
Introduce a way to opt into monomorphic (greedy) type inference for generics, preventing them from widening into unions when instantiated with different types.
|
||||||
|
|
||||||
|
## Motivation
|
||||||
|
|
||||||
|
Currently, Luau's type solver allows generic parameters to widen into unions when they are instantiated with multiple types. For example:
|
||||||
|
|
||||||
|
```luau
|
||||||
|
function test<T>(a: T, b: T): T
|
||||||
|
return a
|
||||||
|
end
|
||||||
|
|
||||||
|
local result = test(1, "string") -- inferred type: 1 | "string"
|
||||||
|
```
|
||||||
|
|
||||||
|
This behaviour can be useful in some cases but is undesirable when a function is intended to enforce strict monomorphism. A monomorphic binding would prevent `T` from being instantiated with different types and instead produce a type error if `b` does not match `a`.
|
||||||
|
|
||||||
|
```luau
|
||||||
|
function test<T!>(a: T, b: T)
|
||||||
|
end
|
||||||
|
|
||||||
|
test(1, "string") -- Type error: Expected `number`, got `string`
|
||||||
|
```
|
||||||
|
|
||||||
|
## Design
|
||||||
|
|
||||||
|
This RFC proposes adding some symbol next to each type variable which is a clear opt-in.
|
||||||
|
|
||||||
|
### New Syntax
|
||||||
|
|
||||||
|
The `T!` syntax would indicate a that the type binding for `T` is monomorphic.
|
||||||
|
|
||||||
|
### Type Checking Rules
|
||||||
|
|
||||||
|
1. When a generic parameter is marked as `T!`:
|
||||||
|
- The first instantiation of `T` determines its type
|
||||||
|
- Any subsequent use of `T` in the same context must match this type exactly
|
||||||
|
- If a different type is encountered, a **type error** is raised
|
||||||
|
- `T` will not expand into a union
|
||||||
|
|
||||||
|
2. **Behavior in Unions**
|
||||||
|
- A function or type with `T!` cannot instantiate `T` with a union
|
||||||
|
- If `T` is already a union, it must remain a union as new types cannot be added to it
|
||||||
|
|
||||||
|
## Drawbacks
|
||||||
|
|
||||||
|
- Introduces a new syntax modifier (`T!`), which may lead to a symbol soup but it doesn't seem too shabby next to `T?`.
|
||||||
|
|
||||||
|
## Alternatives
|
||||||
|
|
||||||
|
- **Type Function Constraint**: Provide a `types.monomorphic<T>` function in user-defined type functions to enforce monomorphism dynamically. But this would probably require some way to propagate an `*error-type*` to the user
|
||||||
|
- **Keywords**: Something like `<greedy T>` or `<strict T>` should also be considered if we want to reduce symbols
|
||||||
|
- **Field-specific bindings**: Flip the relationship from being declared per-type-parameter to be set per-function-parameter such as `function test<T>(a: T, b: T!)` which would imply allow both inference behaviours of generics in an uniform syntax
|
Loading…
Add table
Reference in a new issue