日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
Java14發(fā)布了,不使用"class"也能定義類了?還順手要干掉Lombok!

2020年3月17日發(fā)布,Java正式發(fā)布了JDK 14 ,目前已經(jīng)可以開放下載。在JDK 14中,共有16個(gè)新特性,本文主要來介紹其中的一個(gè)特性:JEP 359: Records

創(chuàng)新互聯(lián)成立與2013年,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目成都網(wǎng)站設(shè)計(jì)、網(wǎng)站建設(shè)網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個(gè)夢想脫穎而出為使命,1280元甘南做網(wǎng)站,已為上家服務(wù),為甘南各地企業(yè)和個(gè)人服務(wù),聯(lián)系電話:18982081108

官方吐槽最為致命

早在2019年2月份,Java 語言架構(gòu)師 Brian Goetz,曾經(jīng)寫過一篇文章(http://cr.openjdk.java.net/~briangoetz/amber/datum.html ),詳盡的說明了并吐槽了Java語言,他和很多程序員一樣抱怨“Java太啰嗦”或有太多的“繁文縟節(jié)”,他提到:開發(fā)人員想要創(chuàng)建純數(shù)據(jù)載體類(plain data carriers)通常都必須編寫大量低價(jià)值、重復(fù)的、容易出錯(cuò)的代碼。如:構(gòu)造函數(shù)、getter/setter、equals()、hashCode()以及toString()等。

以至于很多人選擇使用IDE的功能來自動生成這些代碼。還有一些開發(fā)會選擇使用一些第三方類庫,如Lombok等來生成這些方法,從而會導(dǎo)致了令人吃驚的表現(xiàn)(surprising behavior)和糟糕的可調(diào)試性(poor debuggability)。

那么,Brian Goetz 大神提到的純數(shù)據(jù)載體到底指的是什么呢。他舉了一個(gè)簡單的例子: 

 
 
 
 
  1. final class Point {  
  2.         public final int x;  
  3.         public final int y;   
  4.         public Point(int x, int y) {  
  5.             this.x = x;  
  6.             this.y = y;  
  7.         }   
  8.         // state-based implementations of equals, hashCode, toString  
  9.         // nothing else  
  10.     } 

這里面的Piont其實(shí)就是一個(gè)純數(shù)據(jù)載體,他表示一個(gè)"點(diǎn)"中包含x坐標(biāo)和y坐標(biāo),并且只提供了構(gòu)造函數(shù),以及一些equals、hashCode等方法。

于是,BrianGoetz大神提出一種想法,他提到,Java完全可以對于這種純數(shù)據(jù)載體通過另外一種方式表示。

其實(shí)在其他的面向?qū)ο笳Z言中,早就針對這種純數(shù)據(jù)載體有單獨(dú)的定義了,如Scala中的case、Kotlin中的data以及C#中的record。這些定義,盡管在語義上有所不同,但是它們的共同點(diǎn)是類的部分或全部狀態(tài)可以直接在類頭中描述,并且這個(gè)類中只包含了純數(shù)據(jù)而已。

于是,他提出Java中是不是也可以通過如下方式定義一個(gè)純數(shù)據(jù)載體呢? 

 
 
 
 
  1. record Point(int x, int y) { } 

神說要用record,于是就有了

就像大神吐槽的那樣,我們通常需要編寫大量代碼才能使類變得有用。如以下內(nèi)容:

  •  toString()方法
  •  hashCode() and equals()方法
  •  Getter 方法
  •  一個(gè)共有的構(gòu)造函數(shù)

對于這種簡單的類,這些方法通常是無聊的、重復(fù)的,而且是可以很容易地機(jī)械地生成的那種東西(ide通常提供這種功能)。

當(dāng)你閱讀別人的代碼時(shí),可能會更加頭大。例如,別人可能使用IDE生成的hashCode()和equals()來處理類的所有字段,但是如何才能在不檢查實(shí)現(xiàn)的每一行的情況下確定他寫的對呢?如果在重構(gòu)過程中添加了字段而沒有重新生成方法,會發(fā)生什么情況呢?

大神Brian Goetz提出了使用record定義一個(gè)純數(shù)據(jù)載體的想法,于是,Java 14 中便包含了一個(gè)新特性:EP 359: Records ,作者正是 Brian Goetz

Records的目標(biāo)是擴(kuò)展Java語言語法,Records為聲明類提供了一種緊湊的語法,用于創(chuàng)建一種類中是“字段,只是字段,除了字段什么都沒有”的類。通過對類做這樣的聲明,編譯器可以通過自動創(chuàng)建所有方法并讓所有字段參與hashCode()等方法。這是JDK 14中的一個(gè)預(yù)覽特性。

一言不合反編譯

Records的用法比較簡單,和定義Java類一樣:  

 
 
 
 
  1. record Person (String firstName, String lastName) {}   

如上,我們定義了一個(gè)Person記錄,其中包含兩個(gè)組件:firstName和lastName,以及一個(gè)空的類體。

那么,這個(gè)東西看上去也是個(gè)語法糖,那他到底是怎么實(shí)現(xiàn)的那?

我們先嘗試對他進(jìn)行編譯,記得使用--enable-preview參數(shù),因?yàn)閞ecords功能目前在JDK 14中還是一個(gè)預(yù)覽(preview)功能。 

 
 
 
 
  1. javac --enable-preview --release 14 Person.java  
  2. Note: Person.java uses preview language features.  
  3. Note: Recompile with -Xlint:preview for details. 

如前所述,Record只是一個(gè)類,其目的是保存和公開數(shù)據(jù)。讓我們看看用javap進(jìn)行反編譯,將會得到以下代碼: 

 
 
 
 
  1. public final class Person extends java.lang.Record {  
  2.       private final String firstName; 
  3.        private final String lastName;  
  4.       public Person(java.lang.String, java.lang.String);  
  5.       public java.lang.String toString();  
  6.       public final int hashCode();  
  7.       public final boolean equals(java.lang.Object);  
  8.       public java.lang.String firstName();  
  9.       public java.lang.String lastName();  
  10.      }   

通過反編譯得到的類,我們可以得到以下信息:

1、生成了一個(gè)final類型的Person類(class),說明這個(gè)類不能再有子類了。

2、這個(gè)類繼承了java.lang.Record類,這個(gè)我們使用enum創(chuàng)建出來的枚舉都默認(rèn)繼承java.lang.Enum有點(diǎn)類似

3、類中有兩個(gè)private final 類型的屬性。所以,record定義的類中的屬性都應(yīng)該是private final類型的。

4、有一個(gè)public的構(gòu)造函數(shù),入?yún)⒕褪莾蓚€(gè)主要的屬性。如果通過字節(jié)碼查看其方法體的話,其內(nèi)容就是以下代碼,你一定很熟悉: 

 
 
 
 
  1. public Person(String firstName, String lastName) {  
  2.         this.firstName = firstName;  
  3.         this.lastName = lastName;  
  4.     }  

5、有兩個(gè)getter方法,分別叫做firstName和lastName。這和JavaBean中定義的命名方式有區(qū)別,或許大神想通過這種方式告訴我們r(jià)ecord定義出來的并不是一個(gè)JavaBean吧。

6、還幫我們自動生成了toString(), hashCode() 和 equals()方法。值得一提的是,這三個(gè)方法依賴invokedynamic來動態(tài)調(diào)用包含隱式實(shí)現(xiàn)的適當(dāng)方法。

還可以這樣玩

前面的例子中,我們簡單的創(chuàng)建了一個(gè)record,那么,record中還能有其他的成員變量和方法嗎?我們來看下。

1、我們不能將實(shí)例字段添加到record中。但是,我們可以添加靜態(tài)字段。    

 
 
 
 
  1. record Person (String firstName, String lastName) {  
  2.         static int x;  
  3.     }   

2、我們可以定義靜態(tài)方法和實(shí)例方法,可以操作對象的狀態(tài)。  

 
 
 
 
  1. record Person (String firstName, String lastName) {  
  2.         static int x;     
  3.         public static void doX(){  
  4.             x++;  
  5.         }     
  6.         public String getFullName(){  
  7.             return firstName + " " + lastName;  
  8.         }  
  9.     } 

3、我們還可以添加構(gòu)造函數(shù)。   

 
 
 
 
  1. record Person (String firstName, String lastName) {  
  2.         static int x;     
  3.         public Person{  
  4.             if(firstName == null){  
  5.                 throw new IllegalArgumentException( "firstName can not be null !");  
  6.             }  
  7.         }     
  8.         public Person(String fullName){  
  9.             this(fullName.split(" ")[0],this(fullName.split(" ")[1])  
  10.         }  
  11.     }   

所以,我們是可以在record中添加靜態(tài)字段/方法的,但是問題是,我們應(yīng)該這么做嗎?

請記住,record推出背后的目標(biāo)是使開發(fā)人員能夠?qū)⑾嚓P(guān)字段作為單個(gè)不可變數(shù)據(jù)項(xiàng)組合在一起,而不需要編寫冗長的代碼。這意味著,每當(dāng)您想要向您的記錄添加更多的字段/方法時(shí),請考慮是否應(yīng)該使用完整的類來代替它。

總結(jié)

record 解決了使用類作為數(shù)據(jù)包裝器的一個(gè)常見問題。純數(shù)據(jù)類從幾行代碼顯著地簡化為一行代碼。

但是,record目前是一種預(yù)覽語言特性,這意味著,盡管它已經(jīng)完全實(shí)現(xiàn),但在JDK中還沒有標(biāo)準(zhǔn)化。

那么問題來了,如果你用上了Java 14之后,你還會使用Lombok嗎?哦不,你可能短時(shí)間內(nèi)都用不上,因?yàn)槟憧赡躂ava 8都還沒用熟~ 


網(wǎng)站欄目:Java14發(fā)布了,不使用"class"也能定義類了?還順手要干掉Lombok!
文章鏈接:http://m.5511xx.com/article/dhijicj.html