使用Java 17中的 record 替代 Lombok 的部分功能
在DD長期更新的Java新特性專欄中,已經(jīng)介紹過Java 16中開始支持的新特性:record的使用:2分鐘學(xué)會(huì)Java中record關(guān)鍵字的用法
之前只是做了介紹,但沒有結(jié)合之前的編碼習(xí)慣或規(guī)范來聊聊未來的應(yīng)用變化。最近正好因?yàn)榛ハ鄏eview一些合作伙伴的代碼,產(chǎn)生了一些討論話題,主要正針對(duì)于有了record之后,其實(shí)之前有些用Lombok的場(chǎng)景,是可以替換掉的。
今天我們就來小小的總結(jié)下,我們可以在哪些地方,利用record來替換Lombok。
Lombok的威力
Lombok是我一直都喜歡使用的工具,因?yàn)樗梢宰屛覀兊拇a變的更加整潔。比如:當(dāng)我們要寫一個(gè)User對(duì)象的時(shí)候,如果不使用Lombok,往往需要寫這么多內(nèi)容:
public class User {
private String username;
private String email;
private int userId;
public User(String username, String email, int userId) {
this.username = username;
this.email = email;
this.userId = userId;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getEmail() {
return email;
}
public void setEmail(String email) {
this.email = email;
}
public int getUserId() {
return userId;
}
public void setUserId(int userId) {
this.userId = userId;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
User user = (User) o;
if (userId != user.userId) return false;
if (username != null ? !username.equals(user.username) : user.username != null) return false;
return email != null ? email.equals(user.email) : user.email == null;
}
@Override
public int hashCode() {
int result = username != null ? username.hashCode() : 0;
result = 31 * result + (email != null ? email.hashCode() : 0);
result = 31 * result + userId;
return result;
}
@Override
public String toString() {
return "User{" +
"username='" + username + '\'' +
", email='" + email + '\'' +
", userId=" + userId +
'}';
}
}
在有了Lombok之后呢,通過使用@Data注解,可以將以上內(nèi)容縮減到只需要下面這幾行即可:
@Data
public class User {
private String username;
private String email;
private int userId;
}
@Data注解涵蓋了@Getter、@Setter、@EqualsAndHashCode 和 @toString,所以一個(gè)注解就可以實(shí)現(xiàn)成員變量的Getter和Setter,equals和hashcode方法的重寫,以及toString的重寫。大大降低了代碼量,讓代碼看上去更加整潔。
Lombok的問題
雖然Lombok可以幫助我們少些很多代碼,但它依然有一些缺點(diǎn),比如:
- Lombok并非Java官方提供,而是第三方依賴,依靠社區(qū)維護(hù)。對(duì)于較新的Java版本通常都會(huì)存在兼容性問題,容易產(chǎn)生一些不可預(yù)知的奇怪錯(cuò)誤。
- IDE的兼容限制,并不是所有的IDE都可以完美兼容Lombok,所以可能也會(huì)因此產(chǎn)生一些奇怪的錯(cuò)誤。
使用record來替代
在之前的Java 新特性:record一文中,已經(jīng)提到過record類可以根據(jù)類的字段自動(dòng)生成:構(gòu)造函數(shù)、equals()、hashCode() 和 toString()。這個(gè)功能就跟上面我們演示的Lombok中的@Data非常類似。
寫法的話也非常簡單,只需要這樣一行即可搞定:
public record UserRecord(String username, String email, int userId) {}
可以看到該代碼的整潔度比Lombok的實(shí)現(xiàn)更加干凈。同時(shí),最關(guān)鍵的一點(diǎn),這是Java原生支持的,不需要引入任何第三方依賴!
record類定義完成了,具體使用的話就跟平時(shí)使用其他類一樣,去創(chuàng)建實(shí)例和調(diào)用方法即可,比如下面這樣:
UserRecord userRecord = new UserRecord("didi", "didi@didispace.com", 35);
System.out.println(userRecord.email());
System.out.println(userRecord.toString());
只是,我們?cè)谑褂玫臅r(shí)候需要了解record自動(dòng)生成的代碼與Lombok的區(qū)別,就能馬上上手。
比如,從上面的例子中我們可以看到一個(gè)區(qū)別:獲取成員變量email的時(shí)候,這里并不想傳統(tǒng)getter那樣以getEmail()的形式生成。
哪些情況替代不了?
record類已經(jīng)很強(qiáng)大,但目前并不能完全替代Lombok。主要原因如下:
- record中定義的成員變量是final類型的,初始化后就不能修改了
- record類不能被繼承,所以也無法進(jìn)一步擴(kuò)展
因此,在用record替代Lombok的時(shí)候,更多用來定義靜態(tài)變量,而不是可能會(huì)變化的實(shí)例變量。但是,由于record中也可以定義函數(shù),所以對(duì)于一些對(duì)成員計(jì)算獲得的內(nèi)容,也可以實(shí)現(xiàn)和使用。
總結(jié)
Lombok和record都可以幫助我們編寫更加整潔的代碼。前者是第三方庫,可能存在一些不可預(yù)知的問題和IDE兼容問題,但功能更加全面和強(qiáng)大;后者屬于Java原生的能力,功能雖弱一些,但用好它也能幫助我們減少很多代碼的編寫,且IDE兼容性更好。