Java 8 到 Java 14,改變了哪些你寫代碼的方式?


Java 8 到 Java 14,改變了哪些你寫代碼的方式?

前幾天,JDK 14 正式發佈了,這次發佈的新版本一共包含了16個新的特性。

其實,從Java8 到 Java14 ,真正的改變了程序員寫代碼的方式的特性並不多,我們這篇文章就來看一下都有哪些。

Lambda表達式

Lambda 表達式是 Java 8 中最重要的一個新特性,Lambda 允許把函數作為一個方法的參數。

lambda 表達式的語法格式如下:

<code>(parameters) -> expression

或

(parameters) ->{ statements; }/<code>

如以下例子:

<code>// 1. 不需要參數,返回值為 5  
() -> 5  

// 2. 接收一個參數(數字類型),返回其2倍的值  
x -> 2 * x  

// 3. 接受2個參數(數字),並返回他們的差值  
(x, y) -> x – y  

// 4. 接收2個int型整數,返回他們的和  
(int x, int y) -> x + y  

// 5. 接受一個 string 對象,並在控制檯打印,不返回任何值(看起來像是返回void)  
(String s) -> System.out.print(s)/<code>

Lambda表達式具有簡潔、容易進行並行計算、是未來的編程趨勢等優點,但同時也會帶來調試困難,新人理解成本高等缺點。

Streams API

除了Lambda 表達式外,Java 8中還引入了Stream API,這使得Java終於進入到函數式編程的行列中來了。

Stream 使用一種類似用 SQL 語句從數據庫查詢數據的直觀方式來提供一種對 Java 集合運算和表達的高階抽象。

Stream API可以極大提高Java程序員的生產力,讓程序員寫出高效率、乾淨、簡潔的代碼。

如下圖,就是通過Stream API對集合進行了一系列的操作:

<code>List strings = Arrays.asList("Hollis", "HollisChuang", "hollis", "Hello", "HelloWorld", "Hollis");

Stream s = strings.stream().filter(string -> string.length()<= 6).map(String::length).sorted().limit(3).distinct();/<code>

而且,Stream還支持並行流,在性能上比傳統的for循環要好很多。(詳細用法:《Java 8中處理集合的優雅姿勢——Stream》)

從Lambda表達式和Stream API問世至今,已經有6年的時間了,相信很多人已經在工作中使用過這些特性了。

雖然對於這兩種語法的使用,很多人持有不同的看法,但是作者還是認為這個功能是十分好用的,只是在日常寫代碼的時候不要過分"炫技"使用超長的流式操作,代碼可讀性不要太低就可以了。

新的日期和時間 API

在Java 8之前,日期時間 API 存在諸多問題,如:Date非線程安全、java.util和java.sql的包中都有日期類、日期類並不提供國際化,沒有時區支持。

所以,Java 8通過發佈新的Date-Time API (JSR 310)來進一步加強對日期與時間的處理。

新的java.time包涵蓋了所有處理日期,時間,日期/時間,時區,時刻(instants),過程(during)與時鐘(clock)的操作。

常見操作如下:

<code>// 獲取當前的日期時間
LocalDateTime currentTime = LocalDateTime.now();
System.out.println("當前時間: " + currentTime);

// 時間比較
LocalDate today = LocalDate.now();
LocalDate date1 = LocalDate.of(2014, 01, 14);
if(date1.equals(today)){} 

// 時間增加
LocalTime time = LocalTime.now();
LocalTime newTime = time.plusHours(2); // adding two hours/<code>

但是說實話,Java8中的時間API作者日常工作中用的比較少,主要是有很多歷史代碼,還是依賴Date等類型,使用新的API就要面臨互相轉換問題。

本地變量類型推斷

在Java 10之前版本中,我們想定義定義局部變量時。我們需要在賦值的左側提供顯式類型,並在賦值的右邊提供實現類型:

<code>MyObject value = new MyObject();/<code>

在Java 10中,提供了本地變量類型推斷的功能,可以通過var聲明變量:

<code>var value = new MyObject();/<code>

本地變量類型推斷將引入“var”關鍵字,而不需要顯式的規範變量的類型。

其實,所謂的本地變量類型推斷,也是Java 10提供給開發者的語法糖。雖然我們在代碼中使用var進行了定義,但是對於虛擬機來說他是不認識這個var的,在java文件編譯成class文件的過程中,會進行解糖,使用變量真正的類型來替代var(我反編譯了Java 10的本地變量類型推斷)

Switch 表達式

在JDK 12中引入了Switch表達式作為預覽特性。並在Java 13中修改了這個特性,引入了yield語句,用於返回值。

而在之後的Java 14中,這一功能正式作為標準功能提供出來。

在以前,我們想要在switch中返回內容,還是比較麻煩的,一般語法如下:

<code>int i;

switch (x) {

    case "1":

        i=1;

        break;

    case "2":

        i=2;

        break;

    default:

        i = x.length();

        break;
}/<code>

在JDK13中使用以下語法:

<code>int i = switch (x) {

    case "1" -> 1;

    case "2" -> 2;

    default -> {

        int len = args[1].length();

        yield len;

    }

};/<code>

或者

<code>int i = switch (x) {

    case "1": yield 1;

    case "2": yield 2;

    default: {

        int len = args[1].length();

        yield len;
    }
};/<code>

在這之後,switch中就多了一個關鍵字用於跳出switch塊了,那就是yield,他用於返回一個值。和return的區別在於:return會直接跳出當前循環或者方法,而yield只會跳出當前switch塊。

Text Blocks

Java 13中提供了一個Text Blocks的預覽特性,並且在Java 14中提供了第二個版本的預覽。

text block,文本塊,是一個多行字符串文字,它避免了對大多數轉義序列的需要,以可預測的方式自動格式化字符串,並在需要時讓開發人員控制格式。

我們以前從外部copy一段文本串到Java中,會被自動轉義,如有一段以下字符串:

<code>

  

      

Hello, world

   /<code>

將其複製到Java的字符串中,會展示成以下內容:

<code>"\n" +

"    \n" +

"        

Hello, world

\n" + " \n" + "\n";/<code>

即被自動進行了轉義,這樣的字符串看起來不是很直觀,在JDK 13中,就可以使用以下語法了:

<code>"""


  

      

Hello, world

""";/<code>

使用"""作為文本塊的開始符合結束符,在其中就可以放置多行的字符串,不需要進行任何轉義。看起來就十分清爽了。

如常見的SQL語句:

<code>String query = """

    SELECT `EMP_ID`, `LAST_NAME` FROM `EMPLOYEE_TB`

    WHERE `CITY` = 'INDIANAPOLIS'

    ORDER BY `EMP_ID`, `LAST_NAME`;

""";/<code>

看起來就比較直觀,清爽了。

Records

Java 14 中便包含了一個新特性:EP 359: Records,

Records的目標是擴展Java語言語法,Records為聲明類提供了一種緊湊的語法,用於創建一種類中是“字段,只是字段,除了字段什麼都沒有”的類。通過對類做這樣的聲明,編譯器可以通過自動創建所有方法並讓所有字段參與hashCode()等方法。這是JDK 14中的一個預覽特性。

使用record關鍵字可以定義一個記錄:

<code>record Person (String firstName, String lastName) {}/<code>

record 解決了使用類作為數據包裝器的一個常見問題。純數據類從幾行代碼顯著地簡化為一行代碼。(詳見:Java 14 發佈了,不使用”class”也能定義類了?還順手要幹掉Lombok!)

總結

以上,就是從Java 8 到 Java 14中,新推出的可能會影響開發人員寫代碼的方式的一些主要特性。

不知道大家有沒有發現,最近幾個版本中推出的一些功能,使得Java和Kotlin等語言越來越像了...

新的這些功能,確實在一定程度上可以簡化一些代碼,使得開發過程中更加高效,但是說實話,還沒有好到足夠吸引廣大開發者拋棄Java 8進行大規模遷移!

還是那句話:版本任你發,我用Java 8;但是新特性我們還是要去了解下的。

[1]: https://www.hollischuang.com/archives/3333

[2]: https://www.hollischuang.com/archives/2187

[3]: https://www.hollischuang.com/archives/4548

轉載公眾號:Hollis,現任阿里巴巴技術專家,個人技術博主

推薦觀看:金三已過,銀四月中,六天吃透這本Java核心寶典,跳槽面試不心慌

疫情下的機遇,阿里直招怒斬"P7"offer,自曝狂啃六遍的面試筆記

愈發火爆的微服務、分佈式,2020年Java程序員還學不會基本淘汰了


分享到:


相關文章: