Name: Lasse R.H. Nielsen
E-mail: lrn@google.com
DEP Proposal Location
Further stakeholders:
- Bob Nystrom - rnystrom@google.com
- Lars Bak - bak@google.com
Allow omitting const
inside a known const expression where it's redundant.
Currently "const" in an expression is a prefix used for some const expressions (literal list or map) and for calling const constructors (const Foo(42)
).
This makes const []
a compile-time constant, and []
without the leading "const" is not a constant.
Since compound compile time constants' sub-expressions (list elements, map keys and values, constructor parameters) must also be compile-time constant expressions, this means that a list of lists must be written const [const [42, 37], const [87, 23]]
.
The inner "const" are required by the language specification, but they don't actually provide any extra information because a non-const list in the same position is not allowed by the language. The extra occurrences of const
are redundant.
This proposal makes these redundant inner const
optional, so you can write: const [[42, 37], [87, 23]]
without extra unnecessary const
prefixes. This solves Issue 4046.
The proposal is to make the "const" prefix optional on list/map/constructor expressions where it's currently mandatory. It introduces the concept of an expression being in a "const context". A const context is introduced by a "const" prefixed list/map/constructor expression, and extends to all sub-expressions (elements, keys, values and parameters). It is also introduced by other syntax productions that require a sub-expression to be a compile-time constant.
Inside a const context, some expressions are interpreted differently than they would be outside of an const context:
- Any list literal
<T>[...]
(with or without type parameter) in a const context is a const list literal, with the same meaning as the currentconst <T>[...]
syntax. - Any map literal
<K,V>{...}
(with or without type parameters) in a const context is a const map literal, with the same meaning as the currentconst <K,V>{...}
. - Any constructor invocation without a
const
, whetherFoo<T>(...)
,Foo<T>.bar(...)
, orprefix.Foo<T>.bar(...)
(with or without type parameters) has the same meaning as the currentconst Foo<T>(...)
,const Foo<T>.bar(...)
andconst prefix.Foo<T>.bar(...)
expressions respectively, and all parameter expressions are also in a const context. Any function call that isn't a constructor invocation is still a compile-time error in a const context.
These rules can be seen as describing an "automatic const insertion strategy" that can be used to convert the new syntax to the existing syntax without changing the meaning of expressions. It allows omitting "const" from list/map/constructor expression when the expression is already required to be a compile-time constant expression.
There is a precedent for the constructor invocation syntax: Annotations use the same syntax prefixed by "@" instead of "const", so "@Foo(42)" is interpreted as "const Foo(42)", as if the invocation was in a const context (except that parameters must still be marked as const). With the "const context" concept introduced, it would be possible to say that annotations are always in a const context introduced by the "@".
Other source locations that only allow compile time constant expression, could also be automatically be in a const context. The const-requiring locations are:
- Annotation expressions.
- The initializer expression of const variable declarations.
- The default values of optional parameters.
- The case expressions of a switch.
- The right-hand sides of initializer list entries of const constructors.
The proposal is backwards compatible. Existing valid code will still be valid and have the same meaning.
In all cases where the "const" prefix is made optional, omitting it would currently be a compile time error.
Parsing will be complicated - a const-less const constructor invocation with two (or more) type parameters may be syntactically indistinguishable from two comparison expressions:
const [Foo<int, int>(42)]
This is the same problem that the current proposal for generic functions is suffering from, and it should be solved in the same way (if that is solved).
If we can't find a solution, we should probably not make the const
optional on expressions like this, only where there are no parsing ambiguities.
The Dart specification must be updated to allow const constructor invocations without the initial const prefix.
It may be easier to have a separate limited grammar for const expressions, instead of allowing Foo<int>(42)
everywhere in the grammar, and then declare it a syntax error if the syntax occurs outside of a const context.
If generic functions are introduced with the same syntax, this problem goes away since the syntax will be valid everywhere.
Implementing this proposal requires changes to all Dart parsers to match the changes in the specification.
In most cases, the necessary changes should be restricted to the parser. Further compiler steps should treat the const-less expression as if it has a prefixed const
, and work normally on that.
The initializer lists of const constructors can be treated specially because they can be used both in a compile-time constant way and at runtime.
You could say that the initializer expressions are not in a "const context" but in a "potentially const context", because they allow potentially compile-time constant expressions (which includes some uses of the function parameters).
In a potentially const context, an omitted "const" is fixed differently depending on whether the constructor is being called using new
or const
.
- When a
const
is omitted, the expression will create a compile-time constant when the constructor is called usingconst
, inserting theconst
as above. - When the same constructor is called using
new
, the expression is treated as if it was prefixed bynew
(for constructors) or nothing (for list/map literals). - The sub-expressions of such an expression will themselves be in a potentially const context, and may be potentially compile-time constant expressions, so they can use the constructor parameters.
If the const
is not omitted, the expression is a normal compile-time constant expression, and its sub-expressions are in a const context, not a potentially const context.
There is a number of issues related to this: Issue 20962, Issue 22329
Even without this improvement, the proposal still makes sense.
However, not including this feature will make it a breaking change to add it later using the same syntax. The constructor:
const Foo(x) : this.y = Bar(2, 4);
will be equivalent to:
const Foo(x) : this.y = const Bar(2, 4);
If we later add the "auto new/const" feature, the former constructor would now create different objects when called with new
, so:
identical(new Foo(0).y, new Foo(0).y)
would change from true
to false
.
Using an explicit auto
keyword could avoid that problem.
Top-level and static functions are compile-time constants, but function expressions are not (see Issue 4596). It would be possible to make some function expressions in a const context be compile-time constants (only if they don't refer to any non-static variables - basically if they can be converted to a top-level/static function and the expression replaced by a reference to that function). Since a parameter default value is already in const context, you would be able to write a literal expression directly, solving issue 4596. More generally we could allow "const functionExpression" as an expression, and make the "const" optional in a const context.
See also the const function proposal for this feature by itself.
Even without this improvement, the proposal still makes sense.
TC52, the Ecma technical committee working on evolving the open Dart standard, operates under a royalty-free patent policy, RFPP (PDF). This means if the proposal graduates to being sent to TC52, you will have to sign the Ecma TC52 external contributer form and submit it to Ecma.