使用Lombok @Builder注解導(dǎo)致默認(rèn)值無(wú)效的問(wèn)題
@Builder注解導(dǎo)致默認(rèn)值無(wú)效
使用Lombok注解可以極高的簡(jiǎn)化代碼量,比較好用的注解除了@Data之外,還有@Builder這個(gè)注解,它可以讓你很方便的使用builder模式構(gòu)建對(duì)象,但是今天發(fā)現(xiàn)@Builder注解會(huì)把對(duì)象的默認(rèn)值清掉。
像下面這段代碼,會(huì)導(dǎo)致對(duì)象的name屬性變?yōu)閚ull:
public class BuilderTest {
? ? @lombok.Builder
? ? @lombok.Data
? ? private static class Builder {
? ? ? ? private String name = "1232";
? ? }
? ? @Test
? ? public void test() {
? ? ? ? Builder builder = Builder.builder().build();
? ? ? ? System.out.println(builder.getName());
? ? }
}
---打印結(jié)果---
null那么不想讓這個(gè)默認(rèn)值被清除,就只能用另外一個(gè)注解來(lái)對(duì)屬性進(jìn)行設(shè)置:@lombok.Builder.Default
示例代碼如下:
public class BuilderTest {
? ? @lombok.Builder
? ? @lombok.Data
? ? private static class Builder {
? ? ? ? @lombok.Builder.Default
? ? ? ? private String name = "1232";
? ? }
? ? @Test
? ? public void test() {
? ? ? ? Builder builder = Builder.builder().build();
? ? ? ? System.out.println(builder.getName());
? ? }
}
---打印結(jié)果---
1232需要注意的是@lombok.Builder.Default這個(gè)注解是后來(lái)才有的,目前已知的是1.2.X沒有,1.6.X中有這個(gè)注解。
原因分析
使用@Builder注解的代碼如下:
@lombok.Builder
class Example {
? ? private String name = "123";
}Lombok會(huì)生成如下代碼:
class Example {
? ? private String name;
? ? private Example(String name) {
? ? ? ? this.name = name;
? ? }
? ? public static ExampleBuilder builder() {
? ? ? ? return new ExampleBuilder();
? ? }
? ? public static class ExampleBuilder {
? ? ? ? private String name;
? ? ? ? private ExampleBuilder() {}
? ? ? ? public ExampleBuilder name(String name) {
? ? ? ? ? ? this.name = name;
? ? ? ? ? ? return this;
? ? ? ? }
? ? ? ? @java.lang.Override public String toString() {
? ? ? ? ? ? return "Example(name = " + name + ")";
? ? ? ? }
? ? ? ? public Example build() {
? ? ? ? ? ? return new Example(name);
? ? ? ? }
? ? }
}很顯然,ExampleBuilder的name沒有默認(rèn)值,導(dǎo)致build之后的Example的name是Null值。
lombok@Builder忽略屬性默認(rèn)值的坑點(diǎn)
1. 簡(jiǎn)單使用
他這個(gè)@Builder注解,相比之前的編輯器自動(dòng)生成的getter setter的優(yōu)點(diǎn)在哪呢?
看下面的使用例子
package com.lxk.lombok; import com.lxk.model.Bird; import java.util.Date;
/** * 測(cè)試@Builder * * @author LiXuekai on 2019/5/13 */public class BuilderTest {
public static void main(String[] args) {
Bird bird = new Bird("12", "34", new Date(), "紅色皮膚", "巨大無(wú)比", 18, "典韋", "不序列化的字段,是不會(huì)被轉(zhuǎn)json輸出的");
Bird build = Bird.builder().dog1("12").dog2("34").birthday(new Date())
.color("紅色皮膚").size("巨大無(wú)比").age(18)
.name("典韋").deserialize("不序列化的字段,是不會(huì)被轉(zhuǎn)json輸出的").build();
System.out.println(build.toString());
System.out.println(bird.toString());
}}運(yùn)行結(jié)果:

它相比編輯器生成的getter setter的優(yōu)點(diǎn):
1,代碼很清晰,可讀。可以很明確的知道,你的每一個(gè)屬性的設(shè)置,
2,還有就是鏈?zhǔn)奖磉_(dá)式,一串的就給set屬性了,不需要寫n行代碼,之前自動(dòng)生成的getter和setter,都需要寫一個(gè)對(duì)象,然后再一個(gè)個(gè)的去點(diǎn)每一個(gè)屬性的設(shè)置方法,
我例子里面的那個(gè)全參數(shù)構(gòu)造方法,乍一看,看不出來(lái)每個(gè)參數(shù)是啥意思,但是下面的builder就很清楚的知道設(shè)置的都是啥。
代碼使用的model bean的代碼
package com.lxk.model;
import com.alibaba.fastjson.annotation.JSONField;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;import lombok.NoArgsConstructor;
import java.io.Serializable;
import java.util.Date;
/** * 鳥 * * @author LiXuekai on 2018/10/25 */
@Data@NoArgsConstructor@AllArgsConstructor@Builderpublic class Bird implements Serializable {
private static final long serialVersionUID = 1L;
@JSONField(ordinal = 6, name = "內(nèi)容是json字符串的屬性", jsonDirect = true)
private String dog1;
@JSONField(ordinal = 5, name = "內(nèi)容是json字符串的屬性,對(duì)比下差異")
private String dog2;
/** * ordinal,默認(rèn)值為0,不用設(shè)置啦。 */
@JSONField(ordinal = 4, name = "生產(chǎn)日期", format = "yyyy年MM月dd日 HH時(shí)mm分ss秒")
private Date birthday; @JSONField(ordinal = 3, name = "顏色")
private String color; @JSONField(ordinal = 2, name = "體型大小")
private String size; @JSONField(ordinal = 1, name = "年齡")
private int age;
/** * 反序列化false,那么在反序列化的時(shí)候,就不會(huì)把json的值轉(zhuǎn)給對(duì)象的這個(gè)屬性。 */
@JSONField(name = "名稱", deserialize = false)
private String name;
/** * 不序列化此屬性字段,那么在轉(zhuǎn)json的時(shí)候,就不會(huì)在json中出現(xiàn) */
@JSONField(ordinal = 7, name = "不序列化的屬性字段", serialize = false)
private String deserialize;
}2. 默認(rèn)值問(wèn)題
注意:默認(rèn)值的情況。
如果原來(lái)的model bean 的屬性是設(shè)置有默認(rèn)值的,比如下面的age屬性

然后,運(yùn)行結(jié)果

@Builder默認(rèn)是不支持默認(rèn)值設(shè)置,或者說(shuō),自動(dòng)忽略了model里面設(shè)置的默認(rèn)值。
怎么讓他支持默認(rèn)值的設(shè)置。

在有默認(rèn)值的屬性上使用這個(gè)注解:@Builder.Default
要是你的這個(gè)default不存在,那是因?yàn)槟闶褂玫倪@個(gè)Lombok的版本太低了,試著升級(jí)到1.18.4及以上。我使用的是1.18.8版本。

3. 修改屬性值

在model上加上這個(gè)注解參數(shù):@Builder(toBuilder = true)
然后使用的時(shí)候,使用toBuilder()方法拿到builder

這個(gè)給人的感覺,類似String的replace()方法,他不是直接修改原來(lái)的對(duì)象,而是將結(jié)果以新的對(duì)象返回,所以,我們這兒需要個(gè)對(duì)象接收修改之后的返回值。
上面的執(zhí)行結(jié)果,也可以看出來(lái),修改過(guò)之后,原來(lái)的對(duì)象的值并沒有變化,但是新接收的這個(gè)對(duì)象是修改后的值
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
java8使用filter()取出自己所需數(shù)據(jù)
這篇文章主要介紹了java8使用filter()取出自己所需數(shù)據(jù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-05-05
Java利用JSch實(shí)現(xiàn)SSH遠(yuǎn)程操作的技術(shù)指南
在日常開發(fā)中,許多應(yīng)用需要通過(guò) SSH 協(xié)議遠(yuǎn)程連接服務(wù)器來(lái)執(zhí)行命令、上傳或下載文件,JSch是一個(gè)功能強(qiáng)大的 Java 庫(kù),它提供了便捷的接口來(lái)實(shí)現(xiàn) SSH 連接和其他遠(yuǎn)程管理功能,本文將介紹 JSch 的基本功能,并通過(guò)實(shí)際代碼示例幫助您快速上手,需要的朋友可以參考下2025-03-03
Java報(bào)錯(cuò)net.dean.jraw.http.NetworkException異常的原因及解決方法
在開發(fā)涉及網(wǎng)絡(luò)通信的Java應(yīng)用程序時(shí),我們經(jīng)常需要處理各種網(wǎng)絡(luò)異常,net.dean.jraw.http.NetworkException是在使用jRAW庫(kù)時(shí)可能遇到的一個(gè)異常,本文將詳細(xì)探討NetworkException的成因,并提供多種解決方案,需要的朋友可以參考下2024-12-12
Spring Boot中配置定時(shí)任務(wù)、線程池與多線程池執(zhí)行的方法
這篇文章主要給大家介紹了關(guān)于Spring Boot中配置定時(shí)任務(wù)、線程池與多線程池執(zhí)行的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Spring Boot具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-09-09
idea如何debug看springsecurity的過(guò)濾器順序
這篇文章主要介紹了idea如何debug看springsecurity的過(guò)濾器順序,文中通過(guò)圖文結(jié)合的方式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-04-04
使Java的JButton文字隱藏功能的實(shí)現(xiàn)(不隱藏按鈕的前提)
這篇文章主要介紹了使Java的JButton文字隱藏功能的實(shí)現(xiàn)(不隱藏按鈕的前提),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01

