解決Lombok使用@Builder無(wú)法build父類屬性的問(wèn)題
Lombok使用@Builder無(wú)法build父類屬性
問(wèn)題描述
實(shí)體類使用Lombok的@Builder來(lái)實(shí)現(xiàn)Builder模式,但是如果使用了extend繼承,則子類無(wú)法通過(guò)Builder來(lái)Build父類屬性值
解決方案
父類增加@NoArgsConstructor、@AllArgsConstructor注解來(lái)增加無(wú)參和全參構(gòu)造函數(shù)【注:父類屬性不能設(shè)置為private,否則仍然無(wú)法訪問(wèn),父類不允許有@Builder注解,否則會(huì)和子類沖突】
子類增加@NoArgsConstructor來(lái)增加無(wú)參構(gòu)造函數(shù),自定義一個(gè)全參構(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無(wú)法Build父類屬性問(wèn)題
? ? ?*
? ? ?* @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 測(cè)試代碼
?*/
public class Test {
? ? public static void main(String[] args) {
? ? ? ? // 創(chuàng)建一個(gè)子類
? ? ? ? Child child = Child.builder()
? ? ? ? ? ? ? ? .fatherName("父類名稱")
? ? ? ? ? ? ? ? .childName("子類名稱")
? ? ? ? ? ? ? ? .build();
? ? ? ? System.out.println(String.format("父類名稱:%s,子類名稱:%s", child.getFatherName(), child.getChildName()));
? ? }
}輸出結(jié)果:
父類名稱:父類名稱,子類名稱:子類名稱
lombok @Builder注解和build父類屬性問(wèn)題
1、簡(jiǎn)介
通過(guò)@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注解對(duì)類做了什么?
反編譯后的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類中多了一個(gè)名為UserExt.UserExtBuilder的靜態(tài)內(nèi)部類。這個(gè)靜態(tài)類擁有和UserExt類相同的屬性,并且他額外實(shí)現(xiàn)了一些方法:
1.address、age的屬性方法
其實(shí)這些方法和setAttribute十分類似,只是額外返回了實(shí)例本身,這使得它可以使用類似于鏈?zhǔn)秸{(diào)用的寫(xiě)法。
2.build方法
該方法調(diào)用UserExt類的全參構(gòu)造方法來(lái)生成UserExt實(shí)例。
UserExt類還是實(shí)現(xiàn)了builder方法,這個(gè)方法生成一個(gè)空的UserExt.UserExtBuilder實(shí)例。
4、優(yōu)缺點(diǎn)
- 優(yōu)點(diǎn):
寫(xiě)法更優(yōu)雅,不需要太多的set方法設(shè)置屬性。
- 缺點(diǎn):
在生成UserExt實(shí)例之前,先創(chuàng)建了一個(gè)UserExt.UserExtBuilder實(shí)例,其占用了額外的內(nèi)存。并且Java是按值傳遞,我們可以直接修改引用對(duì)象,不用新建一個(gè)對(duì)象再賦值;而B(niǎo)uilder.build()方法每次調(diào)用都會(huì)new一個(gè)實(shí)例出來(lái)。
5、問(wèn)題:@Builder注解不能 build 父類屬性
從反編譯后的UserExt類可以看出,UserExtBuilder并沒(méi)有從父類User繼承來(lái)的屬性:id、name的填充方法。

解決方案:
在子類和父類中都使用@SuperBuilder,去掉@Builder。但從import的包中也能看到@SuperBuilder注解是experimental實(shí)驗(yàn)性的。不知道會(huì)不存存在什么潛在風(fēng)險(xiǎn),慎用。

從反編譯后的User 和UserExt類來(lái)看:
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對(duì)UserBuilder 與 UserExtBuilder做了繼承關(guān)系
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java 使用Scanner類接收從控制臺(tái)輸入的數(shù)據(jù)方式
這篇文章主要介紹了java 使用Scanner類接收從控制臺(tái)輸入的數(shù)據(jù)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-08-08
基于java socket實(shí)現(xiàn) 聊天小程序
這篇文章主要介紹了基于java socket實(shí)現(xiàn) 聊天小程序,代碼分為服務(wù)器和客戶端,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-12-12
springboot中的springSession的存儲(chǔ)和獲取實(shí)現(xiàn)
這篇文章主要介紹了springboot中的springSession的存儲(chǔ)和獲取實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09
Spring IOC源碼之bean的注冊(cè)過(guò)程講解
這篇文章主要介紹了Spring IOC源碼之bean的注冊(cè)過(guò)程講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
SpringBoot DBUnit 單元測(cè)試(小結(jié))
這篇文章主要介紹了SpringBoot DBUnit 單元測(cè)試(小結(jié)),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-09-09
java動(dòng)態(tài)導(dǎo)出excel壓縮成zip下載的方法
這篇文章主要為大家詳細(xì)介紹了java動(dòng)態(tài)導(dǎo)出excel壓縮成zip下載的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07

