阿里巴巴資深技術專家雷卷:值得開發者關注的 Java 8 後時代的語言特性

  • 2019 年 10 月 3 日
  • 筆記

file
作者 | 阿里巴巴資深技術專家  雷卷,GitHub ID @linux-china

導讀:在 Python、JavaScript 等一眾程式語言崛起風靡之際,一代霸主 Java 風采雖不及當年,但仍橫掃了各大程式語言排行榜,依舊是各大企業級應用開發語言中的 NO.1。從 Java 8 之後,Java 引入了很多有用的新語言特性,以及新工具和性能改善。但是仍有非常多的同學在日常開發中沒有切換到 Java 8 的後續版本。本篇文章將側重開發方向,為大家介紹後 Java 8 時代的特性。

首先我們必須承認,Java 8 是一個里程碑式的版本,這個相信大多數Java程式設計師都認同,其中最知名的是 Streams & Lambda ,這讓 Functional Programming 成為可能,讓 Java 煥發新的活力。這也是即便 Oracle 不在支援 Java 8 的更新,各個雲廠商還是積極支援,站點為https://adoptopenjdk.net/,可以讓 Java 8 能繼續保留非常長的時間。

目前非常多的同學日常開發並沒有切換到 Java 8 後續的版本,所以這篇文章,我們打算寫一個後 Java 8 時代的特性,主要是偏向於開發的,不涉及 GC , Compiler , Java Module , Platform 等,如果一一解釋,估計非常長的文章,當然後續可以寫另外文章介紹。下面的這些特性會影響到我們日常的程式碼編寫。

考慮到 Java 13 馬上發布,所以版本覆蓋從 9 到 13 ,與此同時 Java Release 的方式調整,一些特性是在某一版本引入(preview),後續收到回饋後做了非常多的增強和完善,這裡就不一一說明特性是哪個版本的,你可以理解為後Java 8版本後的特性大雜燴。參考資料來源於官方 features 和 pluralsight 上每一個版本的 Java 特性介紹。

var 關鍵字(局部變數類型推導) Local-Variable Type Inference

Java 支援泛型,但是如果類型非常長,你又不是特別關注,你用 var 關鍵字就可以啦,可以讓你程式碼非常簡潔。Java IDE 都非常好地支援 var,不用擔心程式碼提示等問題。

Map<String, List<Map<String,Object>>>  store = new ConcurrentHashMap<String, List<Map<String,Object>>>();          Map<String, List<Map<String,Object>>>  store = new ConcurrentHashMap<>();          Map<String, List<Map<String,Object>>>  store = new ConcurrentHashMap<String, List<Map<String,Object>>>();    //lambda    BiFunction<String, String, String> function1 = (var s1, var s2) -> s1 + s2;          System.out.println(function1.apply(text1, text2));

複製 confd 文件到 bin 目錄下,啟動 confd。

sudo cp bin/confd /usr/local/bin  confd

實際的使用中還有一些小的限制,如 null 賦值問題等,但是這些不是什麼問題,馬上用起來。

ProcessHandle

雖然我們很少在 Java 中調用系統命令,但是偶爾用到也是有的,當然都是ProcessBuilder 。還有一個就是增強的 ProcessHandle ,可以了解其他進程的一些資訊,如獲取所有進程、某一進程的啟動的命令、啟動時間等等。

ProcessHandle ph =  ProcessHandle.of(89810).get();  System.out.println(ph.info());

Collection factory methods

創建 ArrayList , HashSet 還是用 new 方法,有點過時啦,直接使用工廠方法就可以啦。

Set<Integer> ints = Set.of(1, 2, 3);  List<String> strings = List.of("first", "second");

String 類的新 API

這裡沒法一一列舉,說幾個重要的 ,了解後就不需要第三方的 StringUtils 啦。repeat, isEmpty, isBlank, strip, lines, indent, transform, trimIndent, formatted 等。

HTTP 2 支援

當然如果你使用 OkHTTP 3 那就沒有問題,如果你不想引入其他開發包,那麼 Java 已經支援 HTTP 2 啦,程式碼基本也差不多,當然同步和非同步都支援。

HttpClient client = HttpClient.newHttpClient();          HttpRequest req =                  HttpRequest.newBuilder(URI.create("https://httpbin.org/ip"))                          .header("User-Agent", "Java")                          .GET()                          .build();          HttpResponse<String> resp = client.send(req, HttpResponse.BodyHandlers.ofString());          System.out.println(resp.body());

Text Block(JDK 13)

在之前版本,你要有一大段文本,你要對雙引號進行轉換,轉換後非常不適合閱讀,如下:

String jsonText = "{"id": 1, "nick": "leijuan"}";

新的方式 text block:

//language=json    String cleanJsonText = """          {"id": 1, "nick": "leijuan"}""";

簡單多啦,你可以自由寫程式碼,不用擔心各種雙引號轉換的問題,copy分享轉換等。稍等,你為何要在 cleanJsonText 前面添加 //language=json ,這個什麼鬼?這個是 IntelliJ IDEA 的一個特性,你的 text block 還是有語義的,如是一段HMTL、JSON、SQL 等,添加這個後,馬上就程式碼提示啦。一般人我不告訴他 🙂

text block 還有一個小特性就是基本的模板特性支援,你在text block中要引入一些上下文變數,直接 %s ,然後調用 formatted 方法就可以啦。

//language=html      String textBlock = """      <span style="color: green">Hello %s</span>""";      System.out.println(textBlock.formatted(nick));

Switch 提升

Arrow Labels

接入了 "->" switch 箭頭,不需要寫那麼多 break 啦,程式碼如下:

//legacy      switch (DayOfWeek.FRIDAY) {          case MONDAY: {              System.out.println(1);              break;          }          case WEDNESDAY: {              System.out.println(2);              break;          }          default: {              System.out.println("Unknown");          }      }      //Arrow labels      switch (DayOfWeek.FRIDAY) {          case MONDAY, FRIDAY, SUNDAY -> System.out.println(6);          case TUESDAY -> System.out.println(7);          case THURSDAY, SATURDAY -> System.out.println(8);          case WEDNESDAY -> System.out.println(9);      }

Switch Expressions

也就是 switch 可以有返回值啦,程式碼如如下:

//Yielding a value      int i2 = switch (DayOfWeek.FRIDAY) {          case MONDAY, FRIDAY, SUNDAY -> 6;          case TUESDAY -> 7;          case THURSDAY, SATURDAY -> 8;          case WEDNESDAY -> 9;          default -> {              yield 10;          }      };

關鍵字 yield 表示 switch 表達式的返回值。

我想馬上使用這些特性

你說的這麼多,都非常不錯,但是我們線上還是 Java 8 環境,有什麼用?也就看看而已。不用擔心,有人也想到啦。 這個項目,支援將 JDK 12+ 的各種語法能夠透明編譯到 Java 8 的 VM 上,也就是你現在用這些語法特性跑在 Java 8 上完全沒有問題,所以即便是 Java 8 的環境,沒有問題,以上的特性都可以使用。

如何使用?非常簡單。

首先下載最新的 JDK,如 JDK 13 ,然後在依賴中添加 jabel-java-plugin。

<dependency>              <groupId>com.github.bsideup.jabel</groupId>              <artifactId>jabel-javac-plugin</artifactId>              <version>0.2.0</version>   </dependency>

然後調整一下 maven 的 compiler plugin,將 source 設置為你想要的 Java 版本,如 13 , target 和 release 設置為 8 就可以啦。 IntelliJ IDEA 會自動識別,也不需要調整。

<plugin>                  <groupId>org.apache.maven.plugins</groupId>                  <artifactId>maven-compiler-plugin</artifactId>                  <version>3.8.1</version>                  <configuration>                      <source>13</source>                      <target>8</target>                      <release>8</release>                  </configuration>  </plugin>

這樣你就可以愉快地使用介紹的特性啦。

總結

如果有一些特性沒有整理,而且非常有用的,大家回饋一下,如 API 的調整等,方便後續同學參考一下。


掃描下方二維碼添加小助手,與 8000 位雲原生愛好者討論技術趨勢,實戰進階!

進群暗號:公司-崗位-城市

file