Skip to content

Commit 57690bf

Browse files
committed
Type ascription
Closes rust-lang#354
1 parent a6348a5 commit 57690bf

File tree

1 file changed

+174
-0
lines changed

1 file changed

+174
-0
lines changed

text/0000-type-ascription.md

+174
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,174 @@
1+
- Start Date: 2015-2-3
2+
- RFC PR: (leave this empty)
3+
- Rust Issue: (leave this empty)
4+
5+
# Summary
6+
7+
Add type ascription to expressions and patterns.
8+
9+
Type ascription on expression has already been implemented. Type ascription on
10+
patterns can probably wait until post-1.0.
11+
12+
See also discussion on #354 and [issue 10502](https://github.com/rust-lang/rust/issues/10502).
13+
14+
15+
# Motivation
16+
17+
Type inference is imperfect. It is often useful to help type inference by
18+
annotating a sub-expression or sub-pattern with a type. Currently, this is only
19+
possible by extracting the sub-expression into a variable using a `let`
20+
statement and/or giving a type for a whole expression or pattern. This is un-
21+
ergonomic, and sometimes impossible due to lifetime issues. Specifically, a
22+
variable has lifetime of its enclosing scope, but a sub-expression's lifetime is
23+
typically limited to the nearest semi-colon.
24+
25+
Typical use cases are where a function's return type is generic (e.g., collect)
26+
and where we want to force a coercion.
27+
28+
Type ascription can also be used for documentation and debugging - where it is
29+
unclear from the code which type will be inferred, type ascription can be used
30+
to precisely communicate expectations to the compiler or other programmers.
31+
32+
By allowing type ascription in more places, we remove the inconsistency that
33+
type ascription is currently only allowed on top-level patterns.
34+
35+
## Examples:
36+
37+
Generic return type:
38+
39+
```
40+
// Current.
41+
let z = if ... {
42+
let x: Vec<_> = foo.enumerate().collect();
43+
x
44+
} else {
45+
...
46+
};
47+
48+
// With type ascription.
49+
let z = if ... {
50+
foo.enumerate().collect(): Vec<_>
51+
} else {
52+
...
53+
};
54+
```
55+
56+
Coercion:
57+
58+
```
59+
fn foo<T>(a: T, b: T) { ... }
60+
61+
// Current.
62+
let x = [1u32, 2, 4];
63+
let y = [3u32];
64+
...
65+
let x: &[_] = &x;
66+
let y: &[_] = &y;
67+
foo(x, y);
68+
69+
// With type ascription.
70+
let x = [1u32, 2, 4];
71+
let y = [3u32];
72+
...
73+
foo(x: &[_], y: &[_]);
74+
```
75+
76+
In patterns:
77+
78+
```
79+
struct Foo<T> { a: T, b: String }
80+
81+
// Current
82+
fn foo(Foo { a, .. }: Foo<i32>) { ... }
83+
84+
// With type ascription.
85+
fn foo(Foo { a: i32, .. }) { ... }
86+
```
87+
88+
89+
# Detailed design
90+
91+
The syntax of expressions is extended with type ascription:
92+
93+
```
94+
e ::= ... | e: T
95+
```
96+
97+
where `e` is an expression and `T` is a type. Type ascription has the same
98+
precedence as explicit coercions using `as`.
99+
100+
When type checking `e: T`, `e` must have type `T`. The `must have type` test
101+
includes implicit coercions and subtyping, but not explicit coercions. `T` may
102+
be any well-formed type.
103+
104+
At runtime, type ascription is a no-op, unless an implicit coercion was used in
105+
type checking, in which case the dynamic semantics of a type ascription
106+
expression are exactly those of the implicit coercion.
107+
108+
The syntax of sub-patterns is extended to include an optional type ascription.
109+
Old syntax:
110+
111+
```
112+
P ::= SP: T | SP
113+
SP ::= var | 'box' SP | ...
114+
```
115+
116+
where `P` is a pattern, `SP` is a sub-pattern, `T` is a type, and `var` is a
117+
variable name.
118+
119+
New syntax:
120+
121+
```
122+
P ::= SP: T | SP
123+
SP ::= var | 'box' P | ...
124+
```
125+
126+
Type ascription in patterns has the narrowest precedence, e.g., `box x: T` means
127+
`box (x: T)`.
128+
129+
In type checking, if an expression is matched against a pattern, when matching
130+
a sub-pattern the matching sub-expression must have the ascribed type (again,
131+
this check includes subtyping and implicit coercion). Types in patterns play no
132+
role at runtime.
133+
134+
@eddyb has implemented the expressions part of this RFC,
135+
[PR](https://github.com/rust-lang/rust/pull/21836).
136+
137+
138+
# Drawbacks
139+
140+
More syntax, another feature in the language.
141+
142+
Interacts poorly with struct initialisers (changing the syntax for struct
143+
literals has been [discussed and rejected](https://github.com/rust-lang/rfcs/pull/65)
144+
and again in [discuss](http://internals.rust-lang.org/t/replace-point-x-3-y-5-with-point-x-3-y-5/198)).
145+
146+
If we introduce named arguments in the future, then it would make it more
147+
difficult to support the same syntax as field initialisers.
148+
149+
150+
# Alternatives
151+
152+
We could do nothing and force programmers to use temporary variables to specify
153+
a type. However, this is less ergonomic and has problems with scopes/lifetimes.
154+
Patterns can be given a type as a whole rather than annotating a part of the
155+
pattern.
156+
157+
We could allow type ascription in expressions but not patterns. This is a
158+
smaller change and addresses most of the motivation.
159+
160+
Rely on explicit coercions - the current plan [RFC 401](https://github.com/rust-lang/rfcs/blob/master/text/0401-coercions.md)
161+
is to allow explicit coercion to any valid type and to use a customisable lint
162+
for trivial casts (that is, those given by subtyping, including the identity
163+
case). If we allow trivial casts, then we could always use explicit coercions
164+
instead of type ascription. However, we would then lose the distinction between
165+
implicit coercions which are safe and explicit coercions, such as narrowing,
166+
which require more programmer attention. This also does not help with patterns.
167+
168+
169+
# Unresolved questions
170+
171+
Is the suggested precedence correct? Especially for patterns.
172+
173+
Does type ascription on patterns have backwards compatibility issues?
174+

0 commit comments

Comments
 (0)