|
| 1 | +--- |
| 2 | +title: 最通俗易懂的 Java 10 新特性讲解 |
| 3 | +# toc: false |
| 4 | +date: 2020-02-27 08:01:01 |
| 5 | +url: jdk/jdk10-feature |
| 6 | +tags: |
| 7 | + - Java10 |
| 8 | +categories: |
| 9 | + - Java 新特性 |
| 10 | +--- |
| 11 | + |
| 12 | + |
| 13 | + |
| 14 | +自从 `Java 9` 开始,Oracle 调整了 Java 版本的发布策略,不再是之前的 N 年一个大版本,取而代之的是 6 个月一个小版本,三年一个大版本,这样可以让 Java 的最新改变迅速上线,而小版本的维护周期缩短到下个版本发布之前,大版本的维护周期则是 3 年之久。而 10 就是这么一个小版本,因为 Java 的后续版本基本都会包含之前新特性,所以还是把 `Java 10` 带来的改变单独写一写。 |
| 15 | + |
| 16 | +<!-- more --> |
| 17 | + |
| 18 | +## 1. JEP 322 - 基于时间的版本号 |
| 19 | + |
| 20 | +就像上面说的,Java 调整了发布策略,为了适应这种发布节奏,随着改变的还有 Java 版本号的记录方式。 |
| 21 | + |
| 22 | +版本号的新模式是:`$FEATURE.$INTERIM.$UPDATE.$PATCH` |
| 23 | + |
| 24 | +- `$FEATURE` :基于发布版本,如 Java 10 的 10 。 |
| 25 | +- `$INTERIM` :问题修复和功能增强时 + 1,默认是 0 。 |
| 26 | +- `$UPDATE` :在进行兼容更新,修复新功能安全问题时 +1。 |
| 27 | +- `$PATCH` :特殊问题修复时 +1。 |
| 28 | + |
| 29 | +查看自己的 `Java 10 ` 版本。 |
| 30 | + |
| 31 | +```java |
| 32 | +$ java -version |
| 33 | +java version "10.0.1" 2018-04-17 |
| 34 | +Java(TM) SE Runtime Environment 18.3 (build 10.0.1+10) |
| 35 | +Java HotSpot(TM) 64-Bit Server VM 18.3 (build 10.0.1+10, mixed mode) |
| 36 | +``` |
| 37 | + |
| 38 | +## 2. JEP 286 - 局部类型推断 |
| 39 | + |
| 40 | +JEP 286 提案让 Java 增加了局部类型推断(Local-Variable Type Inference)功能,这让 Java 可以像 `Js` 里的 `var` 或者其他语言的 `auto` 一样可以自动推断数据类型。这其实只是一个新的语法糖,底层并没有变化,在编译时就已经把 `var` 转化成具体的数据类型了,但是这样可以减少代码的编写。 |
| 41 | + |
| 42 | +你可以像下面这样使用 `var` 语法。 |
| 43 | + |
| 44 | +```java |
| 45 | +var hashMap = new HashMap<String, String>(); |
| 46 | +hashMap.put("微信","wn8398"); |
| 47 | +var string = "hello java 10"; |
| 48 | +var stream = Stream.of(1, 2, 3, 4); |
| 49 | +var list = new ArrayList<String>(); |
| 50 | +``` |
| 51 | + |
| 52 | +如果你反编译编译后的这段代码,你会发现还是熟悉的代码片段。 |
| 53 | + |
| 54 | +```java |
| 55 | +HashMap<String, String> hashMap = new HashMap(); |
| 56 | +hashMap.put("微信", "wn8398"); |
| 57 | +String string = "hello java 10"; |
| 58 | +Stream<Integer> stream = Stream.of(1, 2, 3, 4); |
| 59 | +ArrayList<String> list = new ArrayList(); |
| 60 | +``` |
| 61 | + |
| 62 | +`var` 看似好用,其实也有很多限制,官方介绍了 `var` 只能用于下面的几种情况。 |
| 63 | + |
| 64 | +1. 仅限带有初始化的程序的局部变量。 |
| 65 | +2. `for` 循环或者`增强for` 循环中。 |
| 66 | +3. `for`循环中的声明。 |
| 67 | + |
| 68 | +下面演示三种使用情况。 |
| 69 | + |
| 70 | +```java |
| 71 | +public static void testVar() { |
| 72 | + // 情况1,没有初始化会报错 |
| 73 | + // var list; |
| 74 | + var list = List.of(1, 2, 3, 4); |
| 75 | + // 情况2 |
| 76 | + for (var integer : list) { |
| 77 | + System.out.println(integer); |
| 78 | + } |
| 79 | + // 情况3 |
| 80 | + for (var i = 0; i < list.size(); i++) { |
| 81 | + System.out.println(list.get(i)); |
| 82 | + } |
| 83 | +} |
| 84 | +``` |
| 85 | + |
| 86 | +尽管对 `var` 的使用场景增加了很多限制,但在实际使用时你还是要注意,就像下面的代码,你可能一眼并不能看出 `result` 的数据类型。 |
| 87 | + |
| 88 | +```java |
| 89 | +var query = "xxx"; |
| 90 | +var result = dbUtil.executeQuery(query); |
| 91 | +``` |
| 92 | + |
| 93 | + |
| 94 | + |
| 95 | +## 3. JEP 317 - 基于 Java 的 JIT 编译器(实验性) |
| 96 | + |
| 97 | +这个功能让基于 Java 开发的 JIT 编译器 `Graal` 结合 `Java 10` 用在 Linux / x64 平台上,这是一个实验性的 JIT 编译器,有人说这也是 `Java 10` 中最具有未来感的引入。Graal 其实在 `Java 9 ` 中就已经引入了,它带来了 Java 中的 AOT (Ahead Of Time)编译,还支持多种语言,如 Js、Python、Ruby、R、以及其他基于 JVM (如 Java、Kotlin)的和基于 LLVM (如 C、C++)的语言。 |
| 98 | + |
| 99 | +想切换到 `Graal` 可以使用下面的 `jvm` 参数。 |
| 100 | + |
| 101 | +```shell |
| 102 | +-XX:+UnlockExperimentalVMOptions -XX:+UseJVMCICompiler |
| 103 | +``` |
| 104 | + |
| 105 | +这里面有一点我觉得很有意思,看这个图。 |
| 106 | + |
| 107 | + |
| 108 | + |
| 109 | +这就很有意思了,`Graal` 是 Java 语言编写的,用 Java 编写的编译器,然后用来将 Java 字节码编译机器代码。 |
| 110 | + |
| 111 | +`Graal` 官网:[https://www.graalvm.org/](https://www.graalvm.org/) |
| 112 | + |
| 113 | +## 4. JEP 310 - 类数据共享 |
| 114 | + |
| 115 | +JVM 启动时有一步是需要在内存中加载类,而如果有多个 jar,加载第一个 jar 的速度是最慢的。这就延长了程序的启动时间,为了减少这个时间,`Java 10` 引入了应用程序类数据共享(CDS)机制,它可以把你想共享的类共享在程序之间,使不同的 Java 进程之间共享这个类来减少这个类占用的空间以及加载速度。 |
| 116 | + |
| 117 | +## 5. JEP 307 - G1 并行全GC |
| 118 | + |
| 119 | +早在 `Java 9` 时就已经引入了 G1 垃圾收集器,G1 的优点很多。而在 `Java 10` 中还是做了小小调整,当 G1 的并发收集线程不能快速的完成全 GC 时,就会自动切换到**并行**收集,这可以减少在最坏情况下的 GC 速度。 |
| 120 | + |
| 121 | +## 6. JEP 314 - Unicode 语言标签扩展 |
| 122 | + |
| 123 | +这个提案让 JDK 实现了最新的 [LDML 规范](http://www.unicode.org/reports/tr35/tr35.html#Locale_Extension_Key_and_Type_Data)中指定的更多的扩展。 |
| 124 | + |
| 125 | +主要增加了下面几个扩展方法。 |
| 126 | + |
| 127 | +```java |
| 128 | +java.time.temporal.WeekFields::of |
| 129 | +java.util.Calendar::{getFirstDayOfWeek,getMinimalDaysInWeek} |
| 130 | +java.util.Currency::getInstance |
| 131 | +java.util.Locale::getDisplayName |
| 132 | +java.util.spi.LocaleNameProvider |
| 133 | +java.text.DateFormat::get*Instance |
| 134 | +java.text.DateFormatSymbols::getInstance |
| 135 | +java.text.DecimalFormatSymbols::getInstance |
| 136 | +java.text.NumberFormat::get*Instance |
| 137 | +java.time.format.DateTimeFormatter::localizedBy |
| 138 | +java.time.format.DateTimeFormatterBuilder::getLocalizedDateTimePattern |
| 139 | +java.time.format.DecimalStyle::of |
| 140 | +``` |
| 141 | + |
| 142 | +尝试一下。 |
| 143 | + |
| 144 | +```java |
| 145 | +Currency chinaCurrency = Currency.getInstance(Locale.CHINA); |
| 146 | +Currency usCurrency = Currency.getInstance(Locale.US); |
| 147 | +System.out.println("本地货币:" + chinaCurrency); |
| 148 | +System.out.println("US.货币:" + usCurrency); |
| 149 | + |
| 150 | +String displayName = Locale.getDefault().getDisplayName(); |
| 151 | +String displayLanguage = Locale.getDefault().getDisplayLanguage(); |
| 152 | +String displayCountry = Locale.getDefault().getDisplayCountry(); |
| 153 | +System.out.println("本地名称:" + displayName); |
| 154 | +System.out.println("本地语言:" + displayLanguage); |
| 155 | +System.out.println("本地国家:" + displayCountry); |
| 156 | +int firstDayOfWeek = Calendar.getInstance().getFirstDayOfWeek(); |
| 157 | +System.out.println("本地每周第一天:" + firstDayOfWeek); |
| 158 | +``` |
| 159 | + |
| 160 | +输出结果。 |
| 161 | + |
| 162 | +```shell |
| 163 | +本地货币:CNY |
| 164 | +US.货币:USD |
| 165 | +本地名称:中文 (中国) |
| 166 | +本地语言:中文 |
| 167 | +本地国家:中国 |
| 168 | +本地每周第一天:1 |
| 169 | +``` |
| 170 | + |
| 171 | +## 7. API 更新 |
| 172 | + |
| 173 | +`Java 10` 删除了部分 API,也增加了一些实用方法。比如可以通过 `Collection.copyOf` 复制得到一个不可改变集合,即使原来的集合元素发生了变化也不会有影响。 |
| 174 | + |
| 175 | +```java |
| 176 | +var list = new ArrayList<String>(); |
| 177 | +list.add("wechat"); |
| 178 | +list.add("wn8398"); |
| 179 | +List<String> copyList = List.copyOf(list); |
| 180 | +list.add("test"); |
| 181 | +System.out.println(copyList); |
| 182 | +// result |
| 183 | +// [wechat, wn8398] |
| 184 | +``` |
| 185 | + |
| 186 | +也为 `Optional` 增加了一个新的方法 `orElseThrow`。调用这个方法也可以获取到 `optional` 中的 `value` , 但是如果 `value` 为 `null` ,就会抛出异常。 |
| 187 | + |
| 188 | +另外在 `Stream` 最后收集数据的时候,`Collectors` 可以直接指定收集的集合为不可变集合,像下面这样。 |
| 189 | + |
| 190 | +```java |
| 191 | +list.stream().collect(Collectors.toUnmodifiableList()); |
| 192 | +list.stream().collect(Collectors.toUnmodifiableSet()); |
| 193 | +``` |
| 194 | + |
| 195 | +## 其他更新 |
| 196 | + |
| 197 | +`Java 10` 的更新内容不止这些,上面只是列举了常用的以及比较有意思的新特性。还有部分更新如: |
| 198 | + |
| 199 | +1. JEP 312:Thread-Local Handshakes,JVM 内部功能,可以提高 JVM 性能。 |
| 200 | +2. JEP 313:删除了 `javah` 工具,说是删除,其实功能已经包含在 `Java 8` 中的 `javac` 里。 |
| 201 | +3. JEP 316:让 JVM 可以在备用的存储设备(如 NV-DIMM)上分配堆内存,而不用更改程序代码。 |
| 202 | +4. JEP 319:在JDK中提供一组默认的根证书颁发机构(CA)证书。 |
| 203 | + |
| 204 | + |
| 205 | + |
| 206 | +文章案例都已经上传到 Github:[niumoo/jdk-feature](https://github.com/niumoo/jdk-feature) |
0 commit comments