天天看點

@builder注解_如何優雅地構造Bean對象?老鳥:小心了,亂用Lombok注解會出錯分享阿裡 P8 進階架構師吐血總結的 《BATJ大廠進階Java必問面試學習視訊》,附送 100G 面試學習視訊文檔正文開始ORM實體類Builder它做了什麼事?Wither

分享阿裡 P8 進階架構師吐血總結的 《BATJ大廠進階Java必問面試學習視訊》,附送 100G 面試學習視訊文檔

阿裡 P8 級進階架構師吐血總結的面試學習視訊, 内容覆寫很廣,分布式緩存、RPC 調用、Zookeeper、消息隊列、分布式搜尋引擎、分布式 session、分庫分表等。

另外,附送 100G 學習、面試視訊文檔喲~

擷取方式:【關注 + 轉發】後,私信我,回複關鍵字【面試】,即可免費無套路擷取哦~

以下是資源的部分目錄以及内容截圖:

@builder注解_如何優雅地構造Bean對象?老鳥:小心了,亂用Lombok注解會出錯分享阿裡 P8 進階架構師吐血總結的 《BATJ大廠進階Java必問面試學習視訊》,附送 100G 面試學習視訊文檔正文開始ORM實體類Builder它做了什麼事?Wither
@builder注解_如何優雅地構造Bean對象?老鳥:小心了,亂用Lombok注解會出錯分享阿裡 P8 進階架構師吐血總結的 《BATJ大廠進階Java必問面試學習視訊》,附送 100G 面試學習視訊文檔正文開始ORM實體類Builder它做了什麼事?Wither
@builder注解_如何優雅地構造Bean對象?老鳥:小心了,亂用Lombok注解會出錯分享阿裡 P8 進階架構師吐血總結的 《BATJ大廠進階Java必問面試學習視訊》,附送 100G 面試學習視訊文檔正文開始ORM實體類Builder它做了什麼事?Wither

重要的事再說一遍,擷取方式:【關注 + 轉發】後,私信我,回複關鍵字【面試】,即可免費無套路擷取哦~

正文開始

使用java編寫代碼,十之八九都是在寫java類,進而建構java對象。lombok之前也說了不少,但使用了這麼多年,感覺還是有很多技巧可以使用的。

毫無疑問,使用lombok,編寫的java代碼很優雅,而使用起來和普通的java編碼方式建立的類毫無二緻。

不過,這樣就滿足了嗎?實際上lombok很多注解,讓這個java類在使用的時候,也可以更優雅。

本文就從ORM實體類、Builder模式工具類、Wither工具類以及Accessors工具類幾個層面對比一下。

首先說明,不同的方式本質上沒有優劣之分,不過在不同的應用場景就會變得很奇妙了。

ORM實體類

當一個java Bean類作為ORM實體類,或者xml、json的映射類時,需要這個類有這幾個特征:

  • 擁有無參構造器
  • 擁有setter方法,用以反序列化;
  • 擁有getter方法,用以序列化。

那麼最簡單的情況就是:

@Datapublic class UserBean{ private Integer id; private String userName;}
           
  • 複習一下,Data 注解相當于裝配了 @Getter @Setter @RequiredArgsConstructor @ToString @EqualsAndHashCode

那麼,作為實體類、或者序列化的Bean類,足夠用了。

Builder

構造器模式,在很多工具類中頻繁的使用。

package com.pollyduan.builder;import lombok.Builder;@Builderpublic class UserBean { private Integer id; private String userName;}
           

它做了什麼事?

  • 它建立了一個private 的全參構造器。也就意味着 無參構造器沒有; 同時也意味着這個類不可以直接構造對象。
  • 它為每一個屬性建立了一個同名的方法用于指派,代替了setter,而該方法的傳回值為對象本身。

使用一下:

UserBean u=UserBean.builder().id(1001).userName("polly").build();System.out.println(u);
           

還不錯,然并卵,由于這個Bean并沒有getter方法,裡邊的資料沒辦法直接使用。光說沒用,繼續執行你會發現輸出是這個東西:com.pollyduan.builder[email protected],連看都看不出是什麼東東。是以,Builder提供了一個種可能性,實際使用還需要更多的東西。

那麼,我們為了測試友善需要添加 @ToString() 注解,就會輸出 UserBean(id=1001, userName=polly)

換一個思路,你可能想,我不添加ToString注解,我把他轉成json輸出:

UserBean u=UserBean.builder().id(1001).userName("polly").build();ObjectMapper mapper=new ObjectMapper();System.out.println(mapper.writeValueAsString(u));
           

很不幸,你會收到下面的異常:

com.fasterxml.jackson.databind.exc.InvalidDefinitionException: No serializer found for class com.pollyduan.builder.UserBean and no properties discovered to create BeanSerializer (to avoid exception, disable SerializationFeature.FAIL_ON_EMPTY_BEANS)
           

看到 no properties discovered 了吧,沒錯,工具類無法找到屬性,因為沒有 getter,那麼我們加上 @Getter 就可以了。

package com.pollyduan.builder;import lombok.Builder;import lombok.Getter;@[email protected] class UserBean { private Integer id; private String userName;}
           

序列化為json可以了,那麼從一個json反序列化為對象呢?

@[email protected]@Setterpublic class UserBean { private Integer id; private String userName;}
           

還是不行,如無意外,會遇到 com.fasterxml.jackson.databind.exc.InvalidDefinitionException: Cannot construct instance ofcom.pollyduan.builder.UserBean(no Creators, like default construct, exist): cannot deserialize from Object value (no delegate- or property-based Creator)

前面已經交代了,光增加 @Setter 還不夠,他還需要一個無參構造器。那麼,下面可以嗎?

package com.pollyduan.builder;import lombok.Builder;import lombok.Data;@[email protected] class UserBean { private Integer id; private String userName;}
           

同樣不行,因為雖然 Data使用的時候可以直接使用無參構造器,但由于 Builder 引入了全參構造器,那麼按照java原生的規則,無參構造器就沒有了。那麼就再加一個無參構造器

@[email protected]@NoArgsConstructor
           

很不幸,Builder又報錯了,它找不到全參構造器了。好吧,最終的效果如下:

@[email protected]@AllArgsConstructor(access = AccessLevel.PRIVATE)@NoArgsConstructor
           
  • 注意全參構造器的通路級别,不要破壞Builder的規則。

更進一步,看如下示例:

package com.pollyduan.builder;import java.util.List;import lombok.AccessLevel;import lombok.AllArgsConstructor;import lombok.Builder;import lombok.Data;import lombok.NoArgsConstructor;@[email protected]@AllArgsConstructor(access = AccessLevel.PRIVATE)@NoArgsConstructorpublic class UserBean { private Integer id; private String userName; private List addresses;}
           

思考一下,這個List 我們也需要在外面new 一個 ArrayList,然後build進去,使用起來并不舒服。lombok提供了另一個注解配合使用,那就是 @Singular,如下:

package com.pollyduan.builder;import java.util.List;import lombok.AccessLevel;import lombok.AllArgsConstructor;import lombok.Builder;import lombok.Data;import lombok.NoArgsConstructor;@[email protected]@AllArgsConstructor(access = AccessLevel.PRIVATE)@NoArgsConstructorpublic class UserBean { private Integer id; private String userName; @Singular private List favorites;}
           

那麼就可以這樣操作這個清單了。

UserBean u = UserBean.builder().id(1001).userName("polly").favorite("music").favorite("movie").build();
           

是不是很友善?同時還提供了一個 clearXXX方法,清空集合。

還有一個小坑,如果我們增加一個example屬性,然後給它一個預設值:

@[email protected]@AllArgsConstructor(access = AccessLevel.PRIVATE)@NoArgsConstructorpublic class UserBean { private Integer id; private String userName; private String example="123456";}
           

測試一下看:

UserBean u = UserBean.builder().id(1001).userName("polly").build();System.out.println(u.toString());
           

輸出結果:UserBean(id=1001, userName=polly, example=null)

咦,不對呀?預設值呢??這要從Builder的原理來解釋,他實際上是分别設定了一套屬性清單的值,然後使用全參構造器建立對象。那麼,預設值在Bean上,不在Builder上,那麼Builder沒指派,它的值就是null,最後把所有屬性都複制給UserBean,進而null覆寫了預設值。

如何讓Builder實體來有預設值呢?隻需要給該字段增加 @Default 注解級可。

package com.pollyduan.builder;import lombok.AccessLevel;import lombok.AllArgsConstructor;import lombok.Builder;import lombok.Builder.Default;import lombok.Data;import lombok.NoArgsConstructor;@[email protected]@AllArgsConstructor(access = AccessLevel.PRIVATE)@NoArgsConstructorpublic class UserBean { private Integer id; private String userName; @Default private String example="123456";}
           

重新執行測試用例,輸出:UserBean(id=1001, userName=polly, example=123456),OK,沒毛病了。

Wither

用wither方式建構對象,這在Objective-C 中比較多見。

适用的場景是,使用幾個必要的參數建構對象,其他參數,動态的拼裝。舉個例子,我們建構一個ApiClient,它的使用者名和密碼是必須的,他的ApiService的位址有一個預設值,然後我們可以自己定制這個位址。

package com.pollyduan.wither;import lombok.AllArgsConstructor;import lombok.experimental.Wither;@[email protected] //WITHER NEED IT.public class ApiClient {private String appId;private String appKey;private String endpoint="http://api.pollyduan.com/myservice";}
           

如何使用呢?

@Testpublic void test1() {ApiClient client1=new ApiClient(null, null,null);System.out.println(client1);Object client2 = client1.withAppId("10001").withAppKey("abcdefg").withEndpoint("http://127.0.0.1/");System.out.println(client2);}
           

預設的使用null去初始化一個對象還是很奇怪的。和 Builder一樣,Wither也是提供了可能性,實際使用還需要調整一下。

我們可以設定一個必選參數的構造器,如下:

package com.pollyduan.wither;import lombok.AllArgsConstructor;import lombok.NonNull;import lombok.RequiredArgsConstructor;import lombok.experimental.Wither;@[email protected]@AllArgsConstructorpublic class ApiClient {@NonNullprivate String appId;@NonNullprivate String appKey;private String endpoint="http://api.pollyduan.com/myservice";}
           

這樣使用時就可以這樣:

@Testpublic void test1() {ApiClient client1=new ApiClient("10001