New candidate JEP: 361: Switch Expressions (Standard)

Brian Goetz brian.goetz at oracle.com
Tue Oct 29 22:09:15 UTC 2019


The asymmetry with respect to totality between switch expressions and 
statements is indeed unfortunate, and one we approached from a number of 
angles, but we are boxed in on one side by compatibility, and on the 
other by semantics.  Expressions _must_ be total, so we have to enforce 
totality for switch expressions (a non-total expression is 
nonsensical.)  On the other hand, we already have switch statements, and 
they are currently not total, and trying to make them total would break 
existing code.  So we don't have a lot of latitude in our defaults.

> JEP 361 [0] only says "Obviously |switch|statements are not required 
> to be exhaustive".
> Was this discussed previously? I feel like I am missing something. 

It flows from the existing definition of switch statements, which are 
not required to be exhaustive, and so the world is full of 
non-exhaustive switch statements that handle some portion of the state 
space (perhaps either picking off the "easy" states, or pruning off the 
impossible ones and throwing an exception), and then continue on.  We 
can't break this code, so it's here to stay.

It is possible that in the future, we'll expose a way to say "total 
statement switch", which would engage the compiler's aid in proving 
exhaustiveness.  We couldn't just spell that `switch`, for reasons 
above, though.

(That this is just one more reason why "expressions are better than 
statements", and we should prefer to program with expressions when we 
can, is not very comforting.)

Your proposal:

> Probably shortsighted proposal:
> Make arrow-switch statements exhaustive (for sealed types and enums at 
> least). 

was not previously considered, but there's a good reason to not like it, 
which is: it undermines orthogonality.  The improvements to switch here 
were several orthogonal (or mostly orthogonal) changes:
  - expression vs statement
  - single-consequence vs multi-consequence case labels
  - multiple case labels on one line

That users are free to mix and match these is a benefit; composing a 
complex feature out of orthogonal simpler features make it easier to 
reason about, and hidden couplings between them would show up as sharp 
edges.  (For example, if arrow-switch statements were exhaustive but 
colon-switch statements were not, this means that the obvious 
refactoring from one to the other has subtly different semantics.)

Yes, we do have a hidden coupling with respect to exhaustiveness; we 
don't like it, but we don't see a cure that isn't far worse than the 
disease.  (Let's not rehash the supposed cure of "make a new syntactic 
form, and leave switch to rot"; this far-worse cure has already been 
discussed to death.)

So, it seems likely the best we can do is provide a way to opt into 
exhaustiveness analysis for statement switches.  This might be a 
modified keyword (e.g., `total-switch`), or some adornment of the 
default clause (`default: unreachable`), or something else.  But the 
urgency to do this NOW is limited, at least until we have more sources 
of exhaustiveness (sealed types) that are relevant to switch (pattern 
support in switch.)

Cheers,
-Brian




> Subject: Re: New candidate JEP: 361: Switch Expressions (Standard)
> Date: Mon, 28 Oct 2019 13:23:33 +0100
> From: Thomas Zimmermann <zimmermann.tho at gmail.com>
> Reply-To: Amber Expert Group Observers 
> <amber-spec-observers at openjdk.java.net>
> To: amber-spec-observers at openjdk.java.net
>
> Hello dear OpenJDK developers!
>
> I have some feedback regarding switch expressions, more precisely the 
> new non-exhaustive arrow-switch statements.
> The original code was in Kotlin, but I recreated the structure in Java 
> and the problem remains the same.
>
> Context:
> An Android app showing the current location on a map. The UI is 
> listening to a stream of update events that cover
> all corner cases of the dynamic environment that is Android:
>
> enum LocationUpdate {
>     SUCCESS, // new location, yay
>     UNKNOWN, // no location fix for whatever reason
>     PERMISSION_DENIED // user revoked permission while wewere 
> listening to GPS :(
> }
>
> (`LocationUpdate` should really be a sealed type, but let's stay with 
> one preview feature for now)
>
> The UI will react to each update as follows:
>
> void onLocationUpdate(LocationUpdate update) {
>     switch (update) {
>         case SUCCESS -> updateLocationMarker();
>         case UNKNOWN -> removeLocationMarker();
>         // woops, forgot to handle permission denied case
>     }
>     doSomethingElse();
> }
>
> I hope you can see my problem: I expected the compiler to help me even 
> when the switch does not produce a value.
> The most obvious work around is pretty terrible (useless variable, 
> yielding arbitrary values):
>
> void onLocationUpdate(LocationUpdate update) {
>     var ignored = switch (update) {
>         case SUCCESS-> {
>             updateLocationMarker();
>             yieldtrue;
>         }
>         case UNKNOWN-> {
>             removeLocationMarker();
>             yieldtrue;
>         }
>         // compiler error, good
>     };
>     doSomethingElse();
> }
>
> Probably shortsighted proposal:
> Make arrow-switch statements exhaustive (for sealed types and enums at 
> least).
> Getting back the non-exhaustiveness if desired seems simple in this case:
>
> switch (update) {
>     case SUCCESS -> updateLocationMarker();
>     case UNKNOWN -> removeLocationMarker();
>     // don't care about the other cases
>     default -> {}
> }
>
> ... whereas getting exhaustiveness from the non-exhaustive switch is 
> hard (see work around above).
>
>
> JEP 361 [0] only says "Obviously |switch|statements are not required 
> to be exhaustive".
> Was this discussed previously? I feel like I am missing something.
>
> As a sidenote, Kotlin is also missing this feature [1], maybe Java can 
> learn from this (IMO) mistake?
>
> [0] https://openjdk.java.net/jeps/361
> [1] https://youtrack.jetbrains.com/issue/KT-12380
>
> Best regards,
> Thomas Zimmermann
>
> P.S.: I'm very excited about the new upcoming features, stellar work 
> everyone!



More information about the amber-spec-observers mailing list