Switch labels (null again), some tweaking
Brian Goetz
brian.goetz at oracle.com
Wed Apr 28 14:31:59 UTC 2021
>>
>> At least, if you want a total type, you can make it explicit using
>> case var x
>> which is always total whatever the type switched upon.
>>
>> So you have two tools to avoid influence at distance in the totality
>> analysis,
>> - for a switch where you can enumerate all the values enums, sealed
>> type (boolean should be in that category too IMO, but we did not
>> agree on that), if you list all the possible cases, a total case is
>> not required.
>
> Without changes to switch, there is nothing which would break if a
> "new case" is added (to the enum, or to the sealed hierarchy). So,
> listing "all possible cases" relies on the assumption that you know
> what "all cases" are. And right now there's no way to declaratively
> say: this switch (statement) is gonna cover them all. Brian answer
> goes a bit in that direction, which is good, for the outermost layer
> (but the problem still exists somewhat at the nested level).
>
It depends on how much sealing there is. Suppose we have sealed types
Container = Box|Bag and Lunch=Soup|Sandwich. If we are switching on a
Container<Lunch>:
switch (lunch) {
case Box(Soup s):
case Box(Sandwich s):
case Bag(Soup s):
case Bag(Sandwich s):
}
This switch is exhaustive because:
- { Soup, Sandwich } is total on on Lunch (with remainder null)
- therefore { Box(Soup), Box(Sandwich) } is total on Box<Lunch> (with
remainder null, Box(null) }
- similarly { Bag(Soup), Bag(Sandwich) } is total on Bag<Lunch> (with
remainder null, Bag(null) }
- { Box, Bag } is total on Container (with remainder null)
- therefore the cases in our switch are exhaustive on Container<Lunch>
If Lunch were not sealed, the compiler would complain, and we'd need
extra cases to get totality on Box and on Bag, which is needed to get
totality on Container:
switch (lunch) {
case Box(Soup s):
case Box(Sandwich s):
case Box(var s):
case Bag(Soup s):
case Bag(Sandwich s):
case Bag(var s):
}
So I think its similar at every level. And you can say `var` or `Lunch`
or `Object`; they're all total in this context.
>> - use "var" so the total case is explicit, "case var x", "case
>> Foo(var x)", etc
> Sure - this is a good solution. One minor annoying thing is that we
> started from a position where we said "var is just for inference" and
> now we're back in a place where, while var is still, conceptually at
> least, about inference, developers will also be using it for totality
> (as inference guarantees that you are always 100% covered).
No, you can also say `case Object x` or `case Foo(Object x)` and be
semantically equivalent. Var is just type inference here; Remi is
saying "if you don't want to think about what types are total, say var,
and you'll be right." But its just inference for "the minimal type that
is total."
More information about the amber-spec-observers
mailing list