Java 12

Java 12

Java 12早在2019319日发布,它不是一个长久支持(LTS)版本。

Switch表达式(JEP 325)

Java 12中,对Switch表达式的写法进行了改进,虽然是一个语法糖的改进,也让Switch的代码编写变得更加优雅。先看一下在Java 12之前的Switch的写法。

// 通过传入月份,输出月份所属的季节
public static void switchJava12Before(String day) {
    switch (day) {
        case "march":
        case "april":
        case "may":
            System.out.println("春天");
            break;
        case "june":
        case "july":
        case "august":
            System.out.println("夏天");
            break;
        case "september":
        case "october":
        case "november":
            System.out.println("秋天");
            break;
        case "december":
        case "january":
        case "february":
            System.out.println("冬天");
            break;
    }
}

上面的例子中,通过传入一个月份,输出这个月份对应的季节。简单的功能却写了大量代码,而且每个操作都需要一个break来防止Case穿透。由于Switch表达式在Java 12中并不是一个正式发布的功能,还处于预览测试阶段,所以想要使用Java 12去编译运行就需要打开功能预览参数,当然,如果你使用的是Java 14以及更高版本,就可以直接跳过这个部分了。

# 编译时
./bin/javac --enable-preview -source 12 ./Xxx.java
# 运行时
./bin/java --enable-preview Xxx

由于Switch存在的上述问题,所以在Java 12中对Switch进行了改进,让其可以使用case L ->的方式进行操作,那么在Java 12中可以怎么编写这段代码呢?

public static void switchJava12(String day) {
    switch (day) {
        case "march", "april", "may"            -> System.out.println("春天");
        case "june", "july", "august"           -> System.out.println("夏天");
        case "september", "october", "november" -> System.out.println("秋天");
        case "december", "january", "february"  -> System.out.println("冬天");
    }
}

通过测试可以得到预期的输出结果。这还不够,在Switch的改进中,还支持了使用Switch的返回值进行赋值。像下面这样:

String season = switch (day) {
    case "march", "april", "may"            -> "春天";
    case "june", "july", "august"           -> "春天";
    case "september", "october", "november" -> "春天";
    case "december", "january", "february"  -> "春天";
    default -> {
      //throw new RuntimeException("day error")
        System.out.println("day error");
        break "day error";
    }
};
System.out.println("当前季节是:" + season);

虽然编写更加简单了,其实这些只不过是语法糖式的更新,编译后和之前并没有太大区别。

文件对比Files.mismatch

Java中对于文件的操作已经在Java 7中进行了一次增强,这次Java 12又带来了文件对比功能。对比两个文件,如果内容一致,会返回-1 ,如果内容不同,会返回不同的字节开始位置。

// 创建两个文件
Path pathA = Files.createFile(Paths.get("a.txt"));
Path pathB = Files.createFile(Paths.get("b.txt"));

// 写入相同内容
Files.write(pathA,"abc".getBytes(), StandardOpenOption.WRITE);
Files.write(pathB,"abc".getBytes(), StandardOpenOption.WRITE);
long mismatch = Files.mismatch(pathA, pathB);
System.out.println(mismatch);

// 追加不同内容
Files.write(pathA,"123".getBytes(), StandardOpenOption.APPEND);
Files.write(pathB,"321".getBytes(), StandardOpenOption.APPEND);
mismatch = Files.mismatch(pathA, pathB);
System.out.println(mismatch);

// 删除创建的文件
pathA.toFile().deleteOnExit();
pathB.toFile().deleteOnExit();

// RESULT
// -1
// 3

Compact Number

简化的数字格式可以直接转换数字显示格式,比如1000 -> 1K,1000000 -> 1M 。

System.out.println("Compact Formatting is:");
NumberFormat upvotes = NumberFormat.getCompactNumberInstance(new Locale("en", "US"), Style.SHORT);

System.out.println(upvotes.format(100));
System.out.println(upvotes.format(1000));
System.out.println(upvotes.format(10000));
System.out.println(upvotes.format(100000));
System.out.println(upvotes.format(1000000));

// 设置小数位数
upvotes.setMaximumFractionDigits(1);
System.out.println(upvotes.format(1234));
System.out.println(upvotes.format(123456));
System.out.println(upvotes.format(12345678));

可以得到输出如下:

100
1K
10K
100K
1M
1.2K
123.5K
12.3M

JVM相关更新

Shenandoah垃圾收集器

Java 12增加了Shenandoah一个低停顿的垃圾收集器,它可以和Java应用程序中的执行线程同时进行,用来收集垃圾进行内容回收,这样就可以让停顿时间更少。

更多关于Shenandoah垃圾收集器的介绍可以查看文档:Shenandoah GC介绍。

ZGC并发类卸载

ZGC垃圾收集器现在支持类卸载,通过卸载不使用的类来释放这些类相关的数据结构,从而减少应用程序的总体占用空间。因为是并发执行,所以不会停止Java应用程序线程的执行,也因此对GC的暂停时间影响微乎其微。默认情况下启用此功能,但可以使用命令行选项禁用-XX:-ClassUnloading

JVM常量API

在包java.lang.invoke.constant中定义了一系列的基于值的符号引用,可以用来描述各种可加载常量。可以更容易的对关键类文件和运行时构建的名义描述进行建模,特别是对那些从常量池中加载的常量,也让开发者可以更简单标准的处理可加载常量。

默认使用类数据共享(CDS)

这已经不是JDK第一次改进CDS(Class Data Sharing) 功能了,CDS可以让JVM在同一台机器或虚拟机上启动多个应用的速度速度大大增加。原理是在启动应用时共享一些类加载信息,这样启动新进程时就可以使用共享的数据。在Java 12之前此功能需要手动开启,Java 12调整为默认开启。

上一页
下一页