18

作为 Java SE 12 的一部分,引入了switch表达式,并且自 Java SE 14 起,它们已被标准化。它们与switch陈述有何不同?

4

3 回答 3

28

switch声明:

if/else if/else语句不同,switch语句可以有许多可能的执行路径。Aswitch适用于原始类型 、、byte和、它们各自的包装类型(、、和)、枚举类型和类型1。虽然语句用于基于值或条件的范围测试表达式,但语句用于仅基于单个值测试表达式。shortcharintByteShortCharacterIntegerStringif-elseswitch

演示

enum PaymentStatus {
    UNPAID, PARTPAID, PAID, DISPUTED, UNKNOWN;
}

public class Main {
    public static void main(String[] args) {
        String message = "";
        PaymentStatus paymentStatus = PaymentStatus.PARTPAID;

        switch (paymentStatus) {
        case UNPAID:
            message = "The order has not been paid yet. Please make the minimum/full amount to procced.";
            break;
        case PARTPAID:
            message = "The order is partially paid. Some features will not be available. Please check the brochure for details.";
            break;
        case PAID:
            message = "The order is fully paid. Please choose the desired items from the menu.";
            break;
        default:
            throw new IllegalStateException("Invalid payment status: " + paymentStatus);
        }
        System.out.println(message);
    }
}

输出:

The order is partially paid. Some features will not be available. Please check the brochure for details.

switch表达式:

该表达式是在 Java SE 12 中引入的。但是,它在 Java SE 12 和 13 中switch仍然是预览功能,最终在 Java SE 14 中得到了标准化。与任何表达式一样switch表达式计算为单个值,并且可以在语句中使用。它还引入了“箭头case”标签,无需break声明以防止失败。从 Java SE 15 开始,支持的数据类型没有变化(在switch上面的声明部分中提到)。

演示

enum PaymentStatus {
    UNPAID, PARTPAID, PAID, DISPUTED, UNKNOWN;
}

public class Main {
    public static void main(String[] args) {
        PaymentStatus paymentStatus = PaymentStatus.PARTPAID;

        String message = switch (paymentStatus) {
        case UNPAID -> "The order has not been paid yet. Please make the minimum/full amount to procced.";
        case PARTPAID -> "The order is partially paid. Some features will not be available. Please check the brochure for details.";
        case PAID -> "The order is fully paid. Please choose the desired items from the menu.";
        default -> throw new IllegalStateException("Invalid payment status: " + paymentStatus);
        };

        System.out.println(message);
    }
}

输出:

The order is partially paid. Some features will not be available. Please check the brochure for details.

带有的switch表达式yield

从 Java SE 13 开始,您可以使用语句而不是箭头运算符 (->) 从表达式yield返回值。switch

演示

enum PaymentStatus {
    UNPAID, PARTPAID, PAID, DISPUTED, UNKNOWN;
}

public class Main {
    public static void main(String[] args) {
        PaymentStatus paymentStatus = PaymentStatus.PARTPAID;

        String message = switch (paymentStatus) {
        case UNPAID:
            yield "The order has not been paid yet. Please make the minimum/full amount to procced.";
        case PARTPAID:
            yield "The order is partially paid. Some features will not be available. Please check the brochure for details.";
        case PAID:
            yield "The order is fully paid. Please choose the desired items from the menu.";
        default:
            throw new IllegalStateException("Invalid payment status: " + paymentStatus);
        };

        System.out.println(message);
    }
}

输出:

The order is partially paid. Some features will not be available. Please check the brochure for details.

1String JDK 7 添加了对的支持

于 2021-01-10T18:50:16.847 回答
11

不错的文案!但我也可以为单个 case 语句添加多个 case 的功能。下面的例子非常做作(有很多更好的方法可以实现)。它对字符串中的元音、数字、辅音和其他字符进行简单的频率计数。

int count[] = new int[4];

String s = "829s2bi9jskj*&@)(so2i2ksso";

for (char c : s.toCharArray()) {
      int i = switch (c) {
                case  'a', 'e', 'i', 'o', 'u' -> 0;
                case  '0', '1', '2', '3', '4', '5', '6', '7', '8', '9' -> 1;
                case  'b', 'c', 'd', 'f', 'g', 'h', 'j', 'k', 'l',
                      'm', 'n', 'p', 'q', 'r', 's', 't', 'v', 'w',
                      'x', 'y', 'z' -> 2;
                default -> 3;
      };
      count[i]++;
}
System.out.printf("vowels  - %d%n", count[0]);
System.out.printf("digits  - %d%n", count[1]);
System.out.printf("consonants - %d%n", count[2]);
System.out.printf("other   - %d%n", count[3]);

印刷

vowels  - 4
digits  - 7
consonants - 10
other   - 5
于 2021-01-11T02:21:41.810 回答
2

添加到现有答案:yield也可以与一起使用->,其主要目的是在单个表达式不足以满足给定情况时允许使用块:

var test = switch (value) {
    case A -> 1;
    case B -> 2;
    case C -> {
        System.err.println("neither A nor B"); // or some calculation
        yield -1;
    }
}

我还要提到JEP-354,其中提出并描述了 switch 表达式。
可以在Java 语言规范中找到正式规范

于 2021-03-28T22:41:23.263 回答