解決Lombok使用@Builder無法build父類屬性的問題
Lombok使用@Builder無法build父類屬性
問題描述
實(shí)體類使用Lombok的@Builder來實(shí)現(xiàn)Builder模式,但是如果使用了extend繼承,則子類無法通過Builder來Build父類屬性值
解決方案
父類增加@NoArgsConstructor、@AllArgsConstructor注解來增加無參和全參構(gòu)造函數(shù)【注:父類屬性不能設(shè)置為private,否則仍然無法訪問,父類不允許有@Builder注解,否則會和子類沖突】
子類增加@NoArgsConstructor來增加無參構(gòu)造函數(shù),自定義一個全參構(gòu)造函數(shù)(包含父類屬性)
使用示例
父類:
package com.baijia.uqun.individual.management.facade; import lombok.AllArgsConstructor; import lombok.Getter; import lombok.NoArgsConstructor; import lombok.Setter; /** ?* @Author C.W ?* @Date 2020/6/15 5:25 下午 ?* @Description 父類 ?*/ @Setter @Getter @NoArgsConstructor @AllArgsConstructor public class Father { ? ? /** ? ? ?* 父類名稱 ? ? ?*/ ? ? public String fatherName; }
子類:
package com.baijia.uqun.individual.management.facade; import lombok.Builder; import lombok.Getter; import lombok.NoArgsConstructor; import lombok.Setter; /** ?* @Author C.W ?* @Date 2020/6/15 5:26 下午 ?* @Description 子類 ?*/ @Setter @Getter @NoArgsConstructor public class Child extends Father { ? ? /** ? ? ?* 子類名稱 ? ? ?*/ ? ? private String childName; ? ? /** ? ? ?* 用于解決Lombok的Builder無法Build父類屬性問題 ? ? ?* ? ? ?* @param fatherName ? ? ?* @param childName ? ? ?*/ ? ? @Builder(toBuilder = true) ? ? public Child(String fatherName, String childName) { ? ? ? ? super(fatherName); ? ? ? ? this.childName = childName; ? ? } }
使用:
package com.baijia.uqun.individual.management.facade; /** ?* @Author C.W ?* @Date 2020/6/15 5:28 下午 ?* @Description 測試代碼 ?*/ public class Test { ? ? public static void main(String[] args) { ? ? ? ? // 創(chuàng)建一個子類 ? ? ? ? Child child = Child.builder() ? ? ? ? ? ? ? ? .fatherName("父類名稱") ? ? ? ? ? ? ? ? .childName("子類名稱") ? ? ? ? ? ? ? ? .build(); ? ? ? ? System.out.println(String.format("父類名稱:%s,子類名稱:%s", child.getFatherName(), child.getChildName())); ? ? } }
輸出結(jié)果:
父類名稱:父類名稱,子類名稱:子類名稱
lombok @Builder注解和build父類屬性問題
1、簡介
通過@Builder注解,lombok可以方便的實(shí)踐建造者模式。
2、使用
1)創(chuàng)建基類User
import lombok.*; /** * @author Saint */ @Getter @Setter @ToString @NoArgsConstructor @AllArgsConstructor @Builder public class User { private Long id; private String name; }
2)創(chuàng)建子類UserExt
import lombok.*; /** * @author Saint */ @Getter @Setter @Builder @NoArgsConstructor @AllArgsConstructor @ToString public class UserExt extends User{ private String address; private Integer age; }
3)分別使用Builder創(chuàng)建父類和子類
// 父類 User user = User.builder().id(1L).name("saint").build(); // 子類 UserExt userExt = UserExt.builder().address("南京").age(18).build();
3、@Builder注解對類做了什么?
反編譯后的UserExt.class:
public class UserExt extends User { private String address; private Integer age; public static UserExt.UserExtBuilder builder() { return new UserExt.UserExtBuilder(); } public String getAddress() { return this.address; } public Integer getAge() { return this.age; } public void setAddress(final String address) { this.address = address; } public void setAge(final Integer age) { this.age = age; } public UserExt() { } public UserExt(final String address, final Integer age) { this.address = address; this.age = age; } public String toString() { return "UserExt(address=" + this.getAddress() + ", age=" + this.getAge() + ")"; } public static class UserExtBuilder { private String address; private Integer age; UserExtBuilder() { } public UserExt.UserExtBuilder address(final String address) { this.address = address; return this; } public UserExt.UserExtBuilder age(final Integer age) { this.age = age; return this; } public UserExt build() { return new UserExt(this.address, this.age); } public String toString() { return "UserExt.UserExtBuilder(address=" + this.address + ", age=" + this.age + ")"; } } }
注解在編譯后使UserExt類中多了一個名為UserExt.UserExtBuilder的靜態(tài)內(nèi)部類。這個靜態(tài)類擁有和UserExt類相同的屬性,并且他額外實(shí)現(xiàn)了一些方法:
1.address、age的屬性方法
其實(shí)這些方法和setAttribute十分類似,只是額外返回了實(shí)例本身,這使得它可以使用類似于鏈?zhǔn)秸{(diào)用的寫法。
2.build方法
該方法調(diào)用UserExt類的全參構(gòu)造方法來生成UserExt實(shí)例。
UserExt類還是實(shí)現(xiàn)了builder方法,這個方法生成一個空的UserExt.UserExtBuilder實(shí)例。
4、優(yōu)缺點(diǎn)
- 優(yōu)點(diǎn):
寫法更優(yōu)雅,不需要太多的set方法設(shè)置屬性。
- 缺點(diǎn):
在生成UserExt實(shí)例之前,先創(chuàng)建了一個UserExt.UserExtBuilder實(shí)例,其占用了額外的內(nèi)存。并且Java是按值傳遞,我們可以直接修改引用對象,不用新建一個對象再賦值;而Builder.build()方法每次調(diào)用都會new一個實(shí)例出來。
5、問題:@Builder注解不能 build 父類屬性
從反編譯后的UserExt類可以看出,UserExtBuilder并沒有從父類User繼承來的屬性:id、name的填充方法。
解決方案:
在子類和父類中都使用@SuperBuilder,去掉@Builder。但從import的包中也能看到@SuperBuilder注解是experimental實(shí)驗(yàn)性的。不知道會不存存在什么潛在風(fēng)險(xiǎn),慎用。
從反編譯后的User 和UserExt類來看:
public abstract static class UserBuilder<C extends User, B extends User.UserBuilder<C, B>> public abstract static class UserExtBuilder<C extends UserExt, B extends UserExt.UserExtBuilder<C, B>> extends UserBuilder<C, B>
lombok對UserBuilder 與 UserExtBuilder做了繼承關(guān)系
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Springboot集成JWT實(shí)現(xiàn)登錄注冊的示例代碼
本文主要介紹了Springboot集成JWT實(shí)現(xiàn)登錄注冊的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-06-06JavaWeb 網(wǎng)上書店 注冊和登陸功能案例詳解
這篇文章主要介紹了JavaWeb 網(wǎng)上書店 注冊和登陸功能,結(jié)合具體案例形式詳細(xì)分析了JavaWeb 網(wǎng)上書店 注冊和登陸功能具體實(shí)現(xiàn)步驟、操作技巧與注意事項(xiàng),需要的朋友可以參考下2019-08-08Windows系統(tǒng)下JDK1.8與JDK11版本切換超詳細(xì)教程
這篇文章主要給大家介紹了關(guān)于Windows系統(tǒng)下JDK1.8與JDK11版本切換的超詳細(xì)教程,我們可以有多個工程項(xiàng)目,用的JDK版本不一樣,這個時候就需要進(jìn)行自由切換JDK版本了,需要的朋友可以參考下2023-07-07SpringBoot實(shí)現(xiàn)本地存儲文件上傳及提供HTTP訪問服務(wù)的方法
這篇文章主要介紹了SpringBoot實(shí)現(xiàn)本地存儲文件上傳及提供HTTP訪問服務(wù),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-08-08@Valid注解的作用及@Valid注解與@Validated的區(qū)別
這篇文章主要介紹了@Valid注解的作用及@Valid注解與@Validated的區(qū)別,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-08-08Java 動態(tài)編譯在項(xiàng)目中的實(shí)踐分享
在 Java 中,動態(tài)編譯是指在運(yùn)行時動態(tài)地編譯 Java 源代碼,生成字節(jié)碼,并加載到 JVM 中執(zhí)行,動態(tài)編譯可以用于實(shí)現(xiàn)動態(tài)代碼生成、動態(tài)加載、插件化等功能,本文將給大家分享一下Java 動態(tài)編譯在項(xiàng)目中的實(shí)踐,感興趣的同學(xué)跟著小編一起來看看吧2023-07-07Java中的FileWriter用法詳解與實(shí)戰(zhàn)記錄
這篇文章主要給大家介紹了關(guān)于Java中FileWriter用法的相關(guān)資料,包括寫入字符數(shù)據(jù)到文件、字符數(shù)組和部分字符寫入、配合BufferedWriter使用等方法,同時也解釋了其與OutputStreamWriter,BufferedWriter的異同特性,適合簡單的文件寫入操作,需要的朋友可以參考下2024-10-10