譯者 | 胥磊
審校 | 梁策 孫淑娟
很長時間以來,Java 都因其冗長而受到一些開發者的詬病。哪怕是最熱衷 Java 的開發者或許也不得不承認,聲明一個只有兩個屬性的 bean 類Java讓人覺得有點可笑。因為如果遵循推薦規范,最終不僅添加了 getter 和 setter方法,還要添加toString, hashcode 和 equals 方法的重寫,最終大塊的樣板文件式的代碼逼得開發者想放棄Java語言。
Java
import java.util.Objects;
public class Car {
private String brand;
private String model;
private int year;
public String getBrand() {
return brand;
}
public void setBrand(String brand) {
this.brand = brand;
}
public String getModel() {
return model;
}
public void setModel(String model) {
this.model = model;
}
public int getYear() {
return year;
}
public void setYear(int year) {
this.year = year;
}
@Override
public String toString() {
return "Car{" +
"brand='" + brand + '\'' +
", model='" + model + '\'' +
", year=" + year +
'}';
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Car car = (Car) o;
return year == car.year && Objects.equals(brand, car.brand) && Objects.equals(model, car.model);
}
@Override
public int hashCode() {
return Objects.hash(brand, model, year);
}
}
幸運的是,Lombok的橫空出世大大減輕了Java開發者的痛苦。但自從有了與其作用相似的Java Record類型,有人可能會問:Record是否可以全面取代Lombok呢?
1.Lombok是什么?
Lombok是一個與開發環境高度集成的 Java 類庫(當然也可以看成一種語法糖),通過注解改進(spice)代碼,它在 Java 社區中被廣泛接受和使用。
使用了Lombok后,我們新建一個名為Car的類是這樣的:
Java
import lombok.Data;
@Data
public class Car {
private String brand;
private String model;
private int year;
}
代碼更加簡潔,同時也不會影響之前版本的任何功能。
2.Java Record是什么?
定義的每一個Java Record類型可以簡單地看做是值對象(Value Object)模式的實現。它本質還是一個 Java 類,其中所有的屬性都是final的。所以在創建對象時所有類屬性都需要傳遞。Java Record是在 Java 14中引入的,它將持續改進,提升類設計。
通過Record新建Car類是這樣的:
Java
public record Car(String brand, String model, int year)
{
與前一個版本對比,改進非常明顯。
下文將分析Lombok的一些特性,并通過和Record進行比較來評估是否可以永久讓Lombok退出歷史舞臺。
3.不可變性
Record默認情況下是不可變的,這意味著所有的類屬性都被隱式的聲明為 final。我們通常認為Record和值對象(Value Objects)很相似,但是它們沒有 setter 方法,所有的值都需要在構造函數中傳遞。Lombok可以使用@Value 注解達成同樣的效果,但也可以使用@Data注解來保持可變性。
Java
import lombok.Value;
@Value
public class Car {
private String brand;
private String model;
private int year;
}
4.Bean公約
Record并不打算遵循 bean的公約,獲取對象的方法不使用 getX 的方式命名,同時也不再提供 setter 方法和無參的構造函數。另一方面,Lombok只需使用@Data注解就可以將一個類輕松轉換為 JavaBean。
5.Builder
Builder構建器模式是改善對象、創建語法很棒的一種設計模式。Lombok為我們提供了@Builder這個很實用的注解,它幫我們實現了所有樣板代碼。到目前為止,Java Record 并不打算提供此類實現。
Java
import lombok.Builder;
@Builder
public class Car {
private String brand;
private String model;
private int year;
public static void main(String[] args) {
Car myCamaro = Car.builder()
.brand("Chevrolet")
.model("Camaro")
.year(2022)
.build();
}
}
6.多fields類
Record只對少量fields的類是友好的。但是,如果再向其中添加10個fields,那么得到的會是一個龐大的構造函數(繁多的入參),隨之而來的還有多參構造函數所帶來的固有的問題(傳參易錯位,方法重載難判斷等)。
Java
public record DetailedCar(
String brand, String model, int year,
String engineCode, String engineType, String requiredFuel,
String fuelSystem, String maxHorsePower, String maxTorque,
float fuelCapacity) {
}
Java
DetailedCar camaroDetailed = new DetailedCar(
"Chevrolet", "Camaro", 2022, "LTG", "Turbocharged",
"Gas I4", "Direct Injection", "275 @ 560", "295 @ 3000-4500",
19.0f);
使用了Lombok,我們就可以決定創建bean類是選擇使用 setter來設置對象的狀態,還是使用builder這種更簡潔的方式來構造實例。唯一需要注意的是,因為其默認不會強制設置所有屬性,所以可能使實例處于屬性不完整狀態。當然@Builder 注解支持我們將類中所有屬性標記為@nonNull,這樣在構建時屬性就是必需的。如果必需屬性缺失設置則會在運行時拋出一個異常,而不是編譯時強制拋出異常。
Java
import lombok.Builder;
import lombok.NonNull;
@Builder
public class DetailedCar {
@NonNull
private String brand;
@NonNull
private String model;
@NonNull
private int year;
@NonNull
private String engineCode;
@NonNull
private String engineType;
@NonNull
private String requiredFuel;
@NonNull
private String fuelSystem;
@NonNull
private String maxHorsePower;
@NonNull
private String maxTorque;
@NonNull
private float fuelCapacity;
public static void main(String[] args) {
DetailedCar camaroIncomplete = DetailedCar.builder()
.brand("Chevrolet")
.model("Camaro")
.year(2022)
.build();
}
}
輸出:
Exception in thread "main" java.lang.NullPointerException: engineCode is marked non-null but is null
7.繼承
到目前為止,Java Record類是不支持繼承的,所以不能通過擴展其他Record類來創建一個新的Record類,這可能是模型設計的一個限制。盡管如此,我們也要認識到組合優于繼承(面向對象設計原則之七)。
Java
@Data
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = true)
public class Car extends MotorVehicle {
private String brand;
private String model;
private int year;
}
8.結論
Record是Java的一個極佳的新特性,它正推動代碼向更簡潔的方向發展,因此應該多多使用。對于提供了眾多功能的Lombok,考慮到Java變更的緩慢速度,要在項目中將其徹底取代似乎還為時尚早。
原文鏈接:https://dzone.com/articles/records-vs-lombok
譯者介紹
胥磊,51CTO社區編輯,某頭部電商技術副總監,關注Java后端開發,技術管理,架構優化,分布式開發等領域。