1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

JavaのSwitchについて

Last updated at Posted at 2024-11-10

Java 17と21(LTE)で switch が大きく変わったが、意外と罠がある & うまく使うと便利なので情報をまとめてみた

Java 17での変更

Java 17 から switch の新しい記法と switch式が導入された
従来の Java の switch文は書く分岐で処理を完了する場合、break の指定が必要だった
あえて break を使わずに条件によって複数の処理を行わせる効率的な書き方もできたが、あまり可読性は高くなく、break の指定漏れによるミスの方が多かった
そのためか、Java 以降に登場したメジャーな言語の複数分岐では基本的にそれぞれの条件が独立しているのが多かった
Java 17 から追加された switch はその流れを汲んでおり、break なしで記載することができる

jshell> switch (i) {
   ...>     case 1 -> System.out.println("one");
   ...>     case 2 -> System.out.println("two");
   ...>     default -> System.out.println("else");
   ...> }
else

jshell> switch (day) {
   ...>     case "MONDAY", "TUESDAY", "WEDNESDAY" -> System.out.println("Weekday");
   ...>     case "SATURDAY", "SUNDAY" -> System.out.println("Weekend");
   ...>     default -> System.out.println("Invalid day");
   ...> }

jshell> 

また、Java 17 から switchを式として利用可能になった

jshell> var str = switch (i) {
   ...>     case 1 -> "one";
   ...>     case 2 -> "two";
   ...>     default -> "else";
   ...> }
str ==> "else"

jshell> var str = switch (i) {
   ...>     case 1: yield "one";
   ...>     case 2: yield "two";
   ...>     default: yield "else";
   ...> }
str ==> "else"

jshell> 

enum

switch式は可読性も優れているが、私が何より気に入っているのは enum を利用する際の静的チェックである
以下のように switch式では網羅性が担保できない場合はビルドエラーになる

jshell> enum Fruits {APPLE, BANANA}

jshell> var f = Fruits.APPLE;
f ==> APPLE

jshell> var str = switch (f) {
   ...>     case APPLE -> "apple";
   ...> }
|  エラー:
|  switch式がすべての可能な入力値をカバーしていません
|  var str = switch (f) {
|            ^-----------...

jshell> 

この挙動は従来の記法でも同じである

jshell> 

jshell> var str = switch (a) {
   ...>     case APPLE: yield "apple";
   ...> }
|  エラー:
|  switch式がすべての可能な入力値をカバーしていません
|  var str = switch (a) {
|            ^----------...

jshell> 

ちなみに default を書くと必ず網羅性が担保されてしまうので、switch式では default は書かない方が無難である

尚、以下のように記法の種類に関係なく、switch文ではビルドエラーにならない

jshell> switch (a) {
   ...>     case APPLE -> System.out.println("apple");
   ...> }
apple

jshell> switch (a) {
   ...>     case APPLE: System.out.println("apple"); break;
   ...> }
apple

jshell> 

…と思っていたのだが、Java 21で少し事情が変わったようだ

Java 21での変更

null 指定

Java 21で null の条件を指定できるようになったのだが、この機能を指定すると switch文でも switch式でも網羅性が静的チェックできるようになったようだ
(ちなみに17ではプレビュー扱いだった)

jshell> switch (a) {
   ...>     case APPLE: System.out.println("apple");
   ...>     case null: System.out.println("null");
   ...> }
|  エラー:
|  switch文がすべての可能な入力値をカバーしていません
|  switch (a) {
|  ^-----------...

jshell> 

これなら基本的にnull指定しておいた方が無難かもしれない

パターンマッチング

Java 21 で switch のパターンマッチングが利用可能になった

jshell> Object o = "str"
o ==> "str"

jshell> switch (o) {
   ...>     case String s: System.out.println("str"); break;
   ...>     case Integer i: System.out.println("int"); break;
   ...>     default: System.out.println("default");
   ...> }
str

jshell> switch (o) {
   ...>     case String s -> System.out.println("str");
   ...>     case Integer i -> System.out.println("int");
   ...>     default -> System.out.println("default");
   ...> }
str

jshell> 

Java 17 で instanceof のパターンマッチングが利用可能になって便利だが、この記法を使うと複数の子クラスへの対応がシンプルになる

sealed class

sealed class(シールクラス)の詳細は割愛するが、それを switch の条件で指定することで enum のように網羅されていない場合はビルドエラーできる

jshell> sealed interface Fruits permits Apple, Banana {}
|  次を変更しました: インタフェース Fruits。しかし、 class Apple, and class Bananaが宣言されるまで、参照できません

jshell> final class Apple implements Fruits { }
|  次を作成しました: クラス Apple。しかし、 class Fruitsが宣言されるまで、参照できません

jshell> final class Banana implements Fruits { }
|  次を作成しました: クラス Banana

jshell> final Fruits f = new Apple()
f ==> Apple@1b2c6ec2

jshell> switch (f) {
   ...>     case Apple a: System.out.println(a); break;
   ...>     case Banana b: System.out.println(b); break;
   ...> }
REPL.$JShell$12$Apple@1b2c6ec2

jshell> 

こちらは enum と違い、null と同様、switch式だけでなく switch文でもビルドエラーになる

jshell> var str = switch (f) {
   ...>     case Apple a -> "apple";
   ...> }
|  エラー:
|  switch式がすべての可能な入力値をカバーしていません
|  var str = switch (f) {
|            ^-----------...

jshell> switch (f) {
   ...>     case Apple a: System.out.println(a); break;
   ...> }
|  エラー:
|  switch文がすべての可能な入力値をカバーしていません
|  switch (f) {
|  ^-----------...

jshell>

また、switchのパターンマッチングの場合、when節を追加して更に条件を絞り込める

jshell> var str = switch (f) {
   ...>     case Apple a when a != null -> "apple";
   ...>     default -> "default";
   ...> }
str ==> "apple"

jshell> var str = switch (f) {
   ...>     case Apple a when a == null -> "apple";
   ...>     default -> "default";
   ...> }
str ==> "default"

jshell> 

まとめ

  • Java 17
    • switch の新構文と switch式が追加された
    • switch式では default を指定しない方が enum の変更時にビルドエラーで気づける
  • Java 21
    • null を条件指定できるようになった
    • パターンマッチングを指定できるようになった
    • null指定やsealed classのパターンマッチを使うことで、switch式や switch文で網羅性をチェックできる
    • switchのパターンマッチでwhen節を指定することで条件を更に絞り込める

※ バージョンはプレビューのものを除く

1
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?