使用java編寫代碼,十之八九都是在寫java類,從而構建java對象。lombok以前也說了很多,但使用了這麼多年,感受仍是有不少技巧可使用的。java
毫無疑問,使用lombok,編寫的java代碼很優雅,而使用起來和普通的java編碼方式建立的類毫無二致。json
不過,這樣就知足了嗎?實際上lombok不少註解,讓這個java類在使用的時候,也能夠更優雅。api
本文就從ORM實體類、Builder模式工具類、Wither工具類以及Accessors工具類幾個層面對比一下。app
首先說明,不一樣的方式本質上沒有優劣之分,不過在不一樣的應用場景就會變得很奇妙了。工具
當一個java Bean類做爲ORM實體類,或者xml、json的映射類時,須要這個類有這幾個特徵:測試
那麼最簡單的狀況就是:ui
@Data public class UserBean{ private Integer id; private String userName; }
那麼,做爲實體類、或者序列化的Bean類,足夠用了。編碼
構造器模式,在不少工具類中頻繁的使用。日誌
package com.pollyduan.builder; import lombok.Builder; @Builder public class UserBean { private Integer id; private String userName; }
它作了什麼事?code
使用一下:
UserBean u=UserBean.builder() .id(1001) .userName("polly") .build(); System.out.println(u);
還不錯,然並卵,因爲這個Bean並無getter方法,裏邊的數據沒辦法直接使用。光說沒用,繼續執行你會發現輸出是這個東西:com.pollyduan.builder.UserBean@20322d26,連看都看不出是什麼東東。所以,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; @Builder @Getter public class UserBean { private Integer id; private String userName; }
序列化爲json能夠了,那麼從一個json反序列化爲對象呢?
@Builder @Getter @Setter public 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; @Builder @Data public class UserBean { private Integer id; private String userName; }
一樣不行,由於雖然 Data使用的時候能夠直接使用無參構造器,但因爲 Builder 引入了全參構造器,那麼按照java原生的規則,無參構造器就沒有了。那麼就再加一個無參構造器
@Builder @Data @NoArgsConstructor
很不幸,Builder又報錯了,它找不到全參構造器了。好吧,最終的效果以下
@Builder @Data @AllArgsConstructor(access = AccessLevel.PRIVATE) @NoArgsConstructor
更進一步,看以下示例:
package com.pollyduan.builder; import java.util.List; import lombok.AccessLevel; import lombok.AllArgsConstructor; import lombok.Builder; import lombok.Data; import lombok.NoArgsConstructor; @Builder @Data @AllArgsConstructor(access = AccessLevel.PRIVATE) @NoArgsConstructor public class UserBean { private Integer id; private String userName; private List<String> 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; @Builder @Data @AllArgsConstructor(access = AccessLevel.PRIVATE) @NoArgsConstructor public class UserBean { private Integer id; private String userName; @Singular private List<String> favorites; }
那麼就能夠這樣操做這個列表了。
UserBean u = UserBean.builder() .id(1001) .userName("polly") .favorite("music") .favorite("movie") .build();
是否是很方便?同時還提供了一個 clearXXX方法,清空集合。
還有一個小坑,若是咱們增長一個example屬性,而後給它一個默認值:
@Builder @Data @AllArgsConstructor(access = AccessLevel.PRIVATE) @NoArgsConstructor public 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; @Builder @Data @AllArgsConstructor(access = AccessLevel.PRIVATE) @NoArgsConstructor public class UserBean { private Integer id; private String userName; @Default private String example="123456"; }
從新執行測試用例,輸出:UserBean(id=1001, userName=polly, example=123456),OK,沒毛病了。
用wither方式構建對象,這在Objective-C 中比較多見。
適用的場景是,使用幾個必要的參數構建對象,其餘參數,動態的拼裝。舉個例子,咱們構建一個ApiClient,它的用戶名和密碼是必須的,他的ApiService的地址有一個默認值,而後咱們能夠本身定製這個地址。
package com.pollyduan.wither; import lombok.AllArgsConstructor; import lombok.experimental.Wither; @Wither @AllArgsConstructor //WITHER NEED IT. public class ApiClient { private String appId; private String appKey; private String endpoint="http://api.pollyduan.com/myservice"; }
如何使用呢?
@Test public 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; @RequiredArgsConstructor @Wither @AllArgsConstructor public class ApiClient { @NonNull private String appId; @NonNull private String appKey; private String endpoint="http://api.pollyduan.com/myservice"; }
這樣使用時就能夠這樣:
@Test public void test1() { ApiClient client1=new ApiClient("10001", "abcdefgh"); System.out.println(client1); Object client2 = client1.withEndpoint("http://127.0.0.1/"); System.out.println(client2); }
是否是優雅了不少?並且實際上使用時也使用鏈式語法:
ApiClient client1=new ApiClient("10001", "abcdefgh") withEndpoint("http://127.0.0.1/");
另外還有一個小細節,前面的示例輸出以下:
com.pollyduan.wither.ApiClient@782830e com.pollyduan.wither.ApiClient@470e2030
這個日誌代表,with() 返回的對象並非原來的對象,而是一個新對象,這很重要。
訪問器模式,是給一個普通的Bean增長一個便捷的訪問器,包括讀和寫。
它有兩種工做模式,fluent和chain,舉例說明:
package com.pollyduan.accessors; import lombok.Data; import lombok.experimental.Accessors; @Accessors(fluent = true) @Data public class UserBean { private Integer id; private String userName; private String password; }
使用代碼:
UserBean u=new UserBean() .id(10001) .userName("polly") .password("123456"); u.userName("Tom"); System.out.println(u.userName());
這和 Builder 相似,但更小巧,並且不影響屬性的讀寫,只不過使用屬性同名字符串代替了getter和setter。
另外一個模式是 chain:
UserBean u=new UserBean() .setId(10001) .setUserName("polly") .setPassword("123456"); u.setUserName("Tom"); System.out.println(u.getUserName());
能夠看得出來,這fluent的區別就在於使用getter和setter。