java之lombok的構(gòu)建者模式Builder中的泛型寫法說(shuō)明
java lombok構(gòu)建者模式Builder的泛型寫法
第一步:編碼
?public static <T>AjaxResult<T> success(T data) {
? ? ? ? return AjaxResult.<T>builder()
? ? ? ? ? ? ? ? .code(SUCCESS_DEFAULT_CODE)
? ? ? ? ? ? ? ? .success(true)
? ? ? ? ? ? ? ? .msg(OK)
? ? ? ? ? ? ? ? .data(data)
? ? ? ? ? ? ? ? .build();
? ? }第二步:使用:
? ? @ApiOperation(value = "查詢XXX")
? ? @PostMapping("/queryXXX")
? ? public AjaxResult<List<QueryXXXVo>> queryXXX() {
? ? ? ? return AjaxResult.success(XXXService.queryXXX());
? ? }第三步:測(cè)試
請(qǐng)自行測(cè)試
lombok的@Builder注解帶來(lái)的兩大坑
1、@Data和@Builder導(dǎo)致無(wú)參構(gòu)造丟失
單獨(dú)使用@Data注解,是會(huì)生成無(wú)參數(shù)構(gòu)造方法。
單獨(dú)使用@Builder注解,發(fā)現(xiàn)生成了全屬性的構(gòu)造方法。
@Data和@Builder一起用:我們發(fā)現(xiàn)沒(méi)有了默認(rèn)的構(gòu)造方法。如果手動(dòng)添加無(wú)參數(shù)構(gòu)造方法或者用@NoArgsConstructor注解都會(huì)報(bào)錯(cuò)!
兩種解決方法:
1、構(gòu)造方法加上@Tolerate 注解,讓lombok假裝它不存在(不感知)。
@Builder
@Data
public class TestLombok {
? ? @Tolerate
? ? TestLombok() {
? ? }
? ? ......
} ? ?2、直接加上這4個(gè)注解
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class TestLombok {
? ? ......
} ? ?2、@Builder注解導(dǎo)致默認(rèn)值無(wú)效
使用Lombok注解可以極高的簡(jiǎn)化代碼量,比較好用的注解除了@Data之外,還有@Builder這個(gè)注解,它可以讓你很方便的使用builder模式構(gòu)建對(duì)象,但是今天發(fā)現(xiàn)@Builder注解會(huì)把對(duì)象的默認(rèn)值清掉。
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class TestLombok {
? ? private String aa = "zzzz";
? ? public static void main(String[] args) {
? ? ? ? TestLombok build = TestLombok.builder().build();
? ? ? ? System.out.println(build);
? ? }
}輸出:TestLombok(aa=null)
解決: 只需要在字段上面加上@Builder.Default注解即可
@Builder.Default private String aa = "zzzz";
3、分析原因
我們使用注解的方式,底層本質(zhì)就是反射幫我們生成了一系列的setter、getter,所以我們直接打開(kāi)編譯后的target包下面的.class文件,上面的所有原因一目了然!
源文件:
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class TestLombok {
? ? private String aa = "zzzz";
? ? public static void main(String[] args) {
? ? ? ? TestLombok build = TestLombok.builder().build();
? ? ? ? System.out.println(build);
? ? }
}對(duì)應(yīng)的class字節(jié)碼:
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
package com.apple.ucar;
public class TestLombok {
? ? private String aa = "zzzz";
? ? public static void main(String[] args) {
? ? ? ? TestLombok build = builder().build();
? ? ? ? System.out.println(build);
? ? }
? ? public static TestLombok.TestLombokBuilder builder() {
? ? ? ? return new TestLombok.TestLombokBuilder();
? ? }
? ? public String getAa() {
? ? ? ? return this.aa;
? ? }
? ? public void setAa(String aa) {
? ? ? ? this.aa = aa;
? ? }
? ? public boolean equals(Object o) {
? ? ? ? if (o == this) {
? ? ? ? ? ? return true;
? ? ? ? } else if (!(o instanceof TestLombok)) {
? ? ? ? ? ? return false;
? ? ? ? } else {
? ? ? ? ? ? TestLombok other = (TestLombok)o;
? ? ? ? ? ? if (!other.canEqual(this)) {
? ? ? ? ? ? ? ? return false;
? ? ? ? ? ? } else {
? ? ? ? ? ? ? ? Object this$aa = this.getAa();
? ? ? ? ? ? ? ? Object other$aa = other.getAa();
? ? ? ? ? ? ? ? if (this$aa == null) {
? ? ? ? ? ? ? ? ? ? if (other$aa != null) {
? ? ? ? ? ? ? ? ? ? ? ? return false;
? ? ? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? } else if (!this$aa.equals(other$aa)) {
? ? ? ? ? ? ? ? ? ? return false;
? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? return true;
? ? ? ? ? ? }
? ? ? ? }
? ? }
? ? protected boolean canEqual(Object other) {
? ? ? ? return other instanceof TestLombok;
? ? }
? ? public int hashCode() {
? ? ? ? int PRIME = true;
? ? ? ? int result = 1;
? ? ? ? Object $aa = this.getAa();
? ? ? ? int result = result * 59 + ($aa == null ? 43 : $aa.hashCode());
? ? ? ? return result;
? ? }
? ? public String toString() {
? ? ? ? return "TestLombok(aa=" + this.getAa() + ")";
? ? }
? ? public TestLombok() {
? ? }
? ? public TestLombok(String aa) {
? ? ? ? this.aa = aa;
? ? }
? ? public static class TestLombokBuilder {
? ? ? ? private String aa;
? ? ? ? TestLombokBuilder() {
? ? ? ? }
? ? ? ? public TestLombok.TestLombokBuilder aa(String aa) {
? ? ? ? ? ? this.aa = aa;
? ? ? ? ? ? return this;
? ? ? ? }
? ? ? ? public TestLombok build() {
? ? ? ? ? ? return new TestLombok(this.aa);
? ? ? ? }
? ? ? ? public String toString() {
? ? ? ? ? ? return "TestLombok.TestLombokBuilder(aa=" + this.aa + ")";
? ? ? ? }
? ? }
}我們想知道@Data、@Builder等注解底層到底做了什么,直接編譯當(dāng)前文件,即可在生成的.class字節(jié)碼文件查看具體代碼便知道了
比如上述第二點(diǎn),采用@Builder的時(shí)候,這個(gè)aa并沒(méi)有默認(rèn)值,所以會(huì)為空?。?/p>
? public TestLombok.TestLombokBuilder aa(String aa) {
? ? ? ? ? ? this.aa = aa;
? ? ? ? ? ? return this;
? ? ? ? }小結(jié):個(gè)人覺(jué)得如果想要使用@Builder,最簡(jiǎn)單的方法就是直接寫上這4個(gè)注解,有默認(rèn)值的話再加上@Builder.Default直接,正常情況下就沒(méi)啥問(wèn)題了!
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class TestLombok {
?? ?@Builder.Default
? ? private String aa = "zzzz";
? ? public static void main(String[] args) {
? ? ? ? TestLombok build = TestLombok.builder().build();
? ? ? ? System.out.println(build);
? ? }
}總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Spring @Lookup深入分析實(shí)現(xiàn)原理
這篇文章主要介紹了Spring @Lookup實(shí)現(xiàn)原理,我們知道在spring容器中單獨(dú)的一個(gè)抽象類是不能成為一個(gè)bean的,那么有沒(méi)有辦法呢?這個(gè)時(shí)候我們可以使用Lookup注解2023-01-01
Java?處理樹形結(jié)構(gòu)數(shù)據(jù)的過(guò)程
這篇文章主要介紹了Java?處理樹形結(jié)構(gòu)數(shù)據(jù)的過(guò)程,本文給大家分析具體實(shí)現(xiàn)過(guò)程,結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-08-08
java調(diào)用遠(yuǎn)程服務(wù)器的shell腳本以及停止的方法實(shí)現(xiàn)
這篇文章主要介紹了java調(diào)遠(yuǎn)程服務(wù)器的shell腳本以及停止的方法實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03
Spring Boot啟動(dòng)及退出加載項(xiàng)的方法
這篇文章主要介紹了Spring Boot啟動(dòng)及退出加載項(xiàng)的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04

