轉載鏈接地址:https://juejin.im/post/5cf3edf7e51d454f71439c79
1. 前言
在目前眾多編程語言中,Java 語言的表現還是搶眼,不論是企業級服務端開發,還是 Andorid 客戶端開發,都是作為開發語言的首選,甚至在大數據開發領域,Java 語言也能占有一席之地,如 Hadoop,Spark,Flink 大數據等。而作為已經誕生 24 年的 Java 相比其他語言來說,編寫起來略顯得冗長和復雜,而為了能極大提升 Java 開發的效率和代碼簡潔性,一個 Java 庫 Lombok 就這樣誕生了。
首先我們還是看下 Lombok 官方的描述:
Project Lombok is a java library that automatically plugs into your editor and build tools, spicing up your java. Never write another getter or equals method again, with one annotation your class has a fully featured builder, Automate your logging variables, and much more.
從上面的說明里我們可以初步認識一下 Lombok,一個作用於編輯器和構建工具的 Java 庫,可以對編寫的 Java 代碼進行增強,比如說不用再寫實體類的 getter
方法,equals
方法而是自動生成,自動生成日志輸出變量等等,減少重復模板的代碼。大概知道了 Lombok 框架提供的功能后,接下來我們就真正使用一下 Lombok 提供的注解,看它是如何幫助我們提高書寫 Java 代碼的簡潔性和效率的。
本文主要內容涉及如下:
- Lombok 插件安裝
- Lombok 常用注解使用
- lombok-actions:
環境支持:
- JDK 8
- SpringBoot 2.1.4
- Maven 3.6.0
2. 正文
2.1 安裝 Lombok
使用 Lombok 之前我們先要在所使用的 IDE 中進行集成安裝,這里以 IDEA 為例,安裝步驟十分簡單:
-
前往
File -> Settings -> Plugin -> Marketplace
,搜索 Lombok -
選擇搜索結果 Lombok ,點擊 Install 安裝。
-
安裝完成后重啟即可。
基於 Eclipse 的 Lombok 插件安裝方法這里就不詳細描述了,官方也給了對應的文檔說明:projectlombok.org/setup/eclip…
在 IDE 安裝了 Lombok 插件后,我們就可以在 pom.xml
文件中添加 Lombok 的依賴進行使用了。
<dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version>1.18.8</version> <scope>provided</scope> </dependency>
注意:
pom
依賴設置 scope 為provided
,是為了讓 Lombok 庫不被打包進程序。
2.2 @Getter/@Setter
通常我們編寫實體類無論多少個字段,都要為其提供 getter
和 setter
方法,如下面的示例類 User.java

我們常會遇到這種情況:某個實體類新增和修改某個字段,我們都需要單獨處理調整,十分麻煩並且重復。這時候如果我們使用 Lombok 提供 @Getter/@Setter
注解就能幫我們省去 getter 和 setter
方法的維護,由 Lombok 對 User
類自動生成 getter
和 setter
方法,兩者最終的字節碼時一樣的,而我們現在在 User.java
上編寫的代碼僅僅 7 行即可:
@Getter @Setter public class User { private Integer id; private String username; private String password; }
然后用測試類 UserTests.java
測試結果如下:
public class UserTests { @Test public void test() { User user = new User(); user.setUsername("one"); user.setPassword("zxc123"); Assert.assertEquals(user.getUsername(), "one"); Assert.assertEquals(user.getPassword(), "zxc123"); } }
@Getter/@Setter
注解不僅可以使用在類上,還可以使用在字段上,這樣就是表示針對該字段自動生成 getter /setter
方法。
@Getter @Setter private String password;
這里該注解使用在類上,還是在字段上的區別就是,如果注解使用在類上,只針對這個類的非靜態字段有效。
需要注意的一點是:如果 @Getter
注解修飾了 boolean
類型的變量,其生成的 getter
方法簽名是 isXXX
形式,而不是 getXXX
形式。
除此之外,@Getter/@Setter
還提供訪問權限控制的屬性 lombok.AccessLevel value()
, 默認為 PUBLIC
,而其他選值都是枚舉類型:MODULE, PROTECTED, PACKAGE, PRIVATE
2.3 @NonNull
顧名思義,@NonNull
用於標記類中不能允許為 null
的字段或者參數上,任何使用該字段的地方都生成空指針判斷代碼,若@NonNull
標記的變量為 null,拋出 NullPointException
(NPE) 異常。比如下面示例代碼:
public class User { private Integer id; private String username; private String password; public User(Integer id, @NonNull String username, @NonNull String password) { this.id = id; this.username = username; this.password = password; } }
使用了 @NonNull
注解之后我們可以獲取到反編譯之后的字節碼信息如下,這就是 Lombok 給我們生成的最終的代碼:
public class User { private Integer id; private String username; private String password; public User(Integer id, @NonNull String username, @NonNull String password) { if (username == null) { throw new NullPointerException("username is marked non-null but is null"); } else if (password == null) { throw new NullPointerException("password is marked non-null but is null"); } else { this.id = id; this.username = username; this.password = password; } } }
2.4 構造器注解
再來看下平時經常會遇見的場景,為實體類編寫構造器方法,Lombok 提供了三個不同構造器注解 @NoArgsConstructor / @AllArgsConstructor / @RequiredArgsConstructor
分別對用不同構造器方法處理方式,接下來就一一描述。
-
@NoArgsConstructor
為實體類生成無參的構造器方法 -
@AllArgsConstructor
為實體類生成除了static
修飾的字段之外帶有各參數的構造器方法。 -
@RequiredArgsConstructor
為實體類生成指定字段的構造器方法,而這些字段需要被final
,或者@NonNull
修飾。@RequiredArgsConstructor public class User3 { private Integer id; private final String username; @NonNull private String password; }
編譯成功后使用構造器方法時就是這樣的效果:
User3 user3 = new User3("user3", "zxc123");
2.5 @ToString
@ToString
會給類自動生成易閱讀的 toString
方法,帶上有所非靜態字段的屬性名稱和值,這樣就十分便於我們日常開發時進行的打印操作。
@Getter @Setter @AllArgsConstructor @ToString public class User2 { private Integer id; private String username; private String password; }
最終編譯成字節碼,反編譯結果如下:
public class User2 { private Integer id; private String username; private String password; // 省去 setter/getter public String toString() { return "User2(id=" + this.getId() + ", username=" + this.getUsername() + ", password=" + this.getPassword() + ")"; } }
另外,注解 @ToString
還支持設置指定哪些字段的日志化輸出,哪些不需要出現在 toString
方法中。使用屬性 @ToString.Exclude
排除不需要在 toString
中出現的字段,使用 @ToString.Include
標記需要出現在 toString
中的字段,具體用法可參見示例:
@Getter @Setter @AllArgsConstructor @ToString public class User2 { @ToString.Exclude private Integer id; @ToString.Include private String username; @ToString.Include private String password; }
打印 User2
對象的日志效果就是:User2(username=user2, password=zcx123)
。
2.6 @EqualsAndHashCode
@EqualsAndHashCode
注解就是用於根據類所擁有的非靜態字段自動重寫 equals
方法和 hashCode 方法,方便我們用於對象間的比較。類似 @ToString
,@EqualsAndHashCode
還可以使用需要作為比較的字段和排除不需要比較的字段,具體用法可以看如下示例:
@Getter @Setter @AllArgsConstructor @ToString @EqualsAndHashCode public class User4 { @EqualsAndHashCode.Exclude private Integer id; @EqualsAndHashCode.Include private String username; @EqualsAndHashCode.Exclude private String password; }
寫完實體類代碼,我們編寫測試方法試下效果:
@Test public void testEqual() { User4 user4 = new User4(1, "user4", "zxc"); User4 user4_2 = new User4(1, "user4", "123"); Assert.assertEquals(user4, user4_2); // ture }
2.7 @Data/@Value
@Data/@Value
注解,提供了更綜合的生成代碼功能,等價於下面幾個注解
@Getter
@Setter
@RequiredArgsConstructor
@ToString
@EqualsAndHashCode
兩個注解都只能使用在類上,與 @Data
不同, @Value
用來修飾不可變的類上。一般實體類沒有特別的限制的話,通常可以直接使用 @Data
注解修飾。
2.8 @Builder
@Builder
是一個非常強大的注解,提供了一種基於建造者模式的構建對象的 API。使用 @Builder
注解為給我們的實體類自動生成 builder()
方法,並且直接根據字段名稱方法進行字段賦值,最后使用 build()
方法構建出一個實體對象。
@Data @Builder public class User6 { private Integer id; private String username; private String password; } @Test public void testBuilder() { User6 user6 = User6.builder().id(1).username("user6").password("zxc123").build(); log.warn("testLog: {}", user6); // User6(id=1, username=user6, password=zxc123) }
需要注意的是 @Builder
不支持父類字段的生成,當一個實體類存在父類時,@Builder
只能生成當前類的字段構建方法。若需要用到父類的字段方法時, Lombok 提供了新的注解 @SuperBuilder
來應對這種情況,下面是 @SuperBuilder
注解的使用方式:
@SuperBuilder @Getter @Setter public class Parent { private int id; private String name; } @SuperBuilder @Data public class Child extends Parent { private String childName; }
調用示例:
Child child = Child.builder().id(1).name("父類名稱").childName("子類名稱").build();
System.out.println(child.getId());
由於 Lombok Plugin 還未更新支持
@SuperBuilder
,所以以上寫法在 IDEA 下還會提示編譯錯誤,無法找到builder()
方法。
也可以參考此文方式去處理繼承的情況:reinhard.codes/2015/09/16/…
2.9 日志注解
正對程序類中常見不同框架 Logger 對象,Lombok 也提供了注解,來自動生成 Logger 對象,實現優雅地輸出日志,只需要在類上使用日志注解如 @Log
。當然 Lombok 支持了多個日志框架,並且提供對應的注解如下:
@CommonsLog 等價效果: private static final org.apache.commons.logging.Log log = org.apache.commons.logging.LogFactory.getLog(LogExample.class); @Flogger 等價效果: private static final com.google.common.flogger.FluentLogger log = com.google.common.flogger.FluentLogger.forEnclosingClass(); @JBosLog 等價效果: private static final org.jboss.logging.Logger log = org.jboss.logging.Logger.getLogger(LogExample.class); @Log 等價效果: private static final java.util.logging.Logger log = java.util.logging.Logger.getLogger(LogExample.class.getName()); @Log4j 等價效果: private static final org.apache.log4j.Logger log = org.apache.log4j.Logger.getLogger(LogExample.class); @Log4j2 等價效果: private static final org.apache.logging.log4j.Logger log = org.apache.logging.log4j.LogManager.getLogger(LogExample.class); @Slf4j 等價效果: private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(LogExample.class); @XSlf4j 等價效果: private static final org.slf4j.ext.XLogger log = org.slf4j.ext.XLoggerFactory.getXLogger(LogExample.class);
下面代碼使用 @Slf4j
注解進行日志輸出:
@Slf4j public class UserTests { // .... @Test public void testLog() { User5 user5 = new User5(); user5.setId(1); user5.setUsername("user5"); user5.setPassword("zxc123"); log.warn("testLog: {}", user5); // 21:57:15.488 [main] WARN com.one.learn.lombok.UserTests - testLog: User5(id=1, username=user5, password=zxc123) } }
2.10 @Cleanup
@Cleanup
用於標記需要釋放清理操作的資源對象變量,如 FileInputStream
, FileOutputStream
等,標記之后資源對象使用完畢后,就會被自動關閉和清理,實際上這里 Lombok 實現效果與 Java7 特性 try with resource
一樣, 為我們屏蔽了關閉資源的模板代碼,下面給出 @Cleanup
的使用示例:
public class CleanupExample { public static void main(String[] args) throws IOException { @Cleanup InputStream in = new FileInputStream(args[0]); @Cleanup OutputStream out = new FileOutputStream(args[1]); byte[] b = new byte[10000]; while (true) { int r = in.read(b); if (r == -1) { break; } out.write(b, 0, r); } } }
將 CleanupExample.java
編譯生成的字節碼反編譯可以得到如下結果:
public class CleanupExample { //... public static void main(String[] args) throws IOException { FileInputStream in = new FileInputStream(args[0]); try { FileOutputStream out = new FileOutputStream(args[1]); try { byte[] b = new byte[10000]; while(true) { int r = in.read(b); if (r == -1) { return; } out.write(b, 0, r); } } finally { if (Collections.singletonList(out).get(0) != null) { out.close(); } } } finally { if (Collections.singletonList(in).get(0) != null) { in.close(); } } } }
2.11 @SneakyThrows
@SneakyThrows
主要用於在沒有 throws
關鍵字的情況下,隱蔽地拋出受檢查異常,為我們平常開發中需要異常拋出時省去的 throw
操作,下面為使用 @SneakyThrows
的示例代碼:
public class SneakyThrowsExample implements Runnable { @SneakyThrows(UnsupportedEncodingException.class) public String utf8ToString(byte[] bytes) { return new String(bytes, "UTF-8"); } @SneakyThrows public void run() { throw new Throwable(); } }
最終編譯成字節碼,反編譯結果如下:
public class SneakyThrowsExample implements Runnable { public SneakyThrowsExample() { } public String utf8ToString(byte[] bytes) { try { return new String(bytes, "UTF-8"); } catch (UnsupportedEncodingException var3) { throw var3; } } public void run() { try { throw new Throwable(); } catch (Throwable var2) { throw var2; } } }
2.12 val/var
val/var
用於局部變量的修飾,有了這注解修飾后,變量的類型就會自動通過等號右邊的表達式推斷出來,這個功能借鑒於許多編程語言的自動類型推斷的特性。 而 val
與 var
的區別在於, val
用於修飾不可變變量,var 修飾可變變量。當 val
修飾的變量被重新賦值時,編譯器就會提示異常:Error: java: 無法為最終變量 X 分配值
。實際用法也比較簡單,可參考下面代碼:
@Slf4j public class VarValExample { public static void main(String[] args) { val text = "abc"; // text = "1"; // Error: java: 無法為最終變量 text 分配值`。 var num = 1; num = 2; log.info("text:{},num:{}", text, num); // text:abc,num:2 } }
3. 結語
到這里我們學習了 Lombok 的近乎 80% 常用的注解,應用在我們的日常開發中已經是綽綽有余了,開始嘗試 使用 Lombok 吧,慢慢地就會感受下效率的提升以及代碼的優雅。
如果讀完覺得有收獲的話,歡迎點【好看】,點擊文章頭圖,掃碼關注【聞人的技術博客】😄😄😄。
4. 參考
-
Project Lombok: projectlombok.org/
-
Project Lombok: Reducing Java Boilerplate Code:stackabuse.com/project-lom…
-
Lombok 常用注解的使用總結:segmentfault.com/a/119000001…
-
Lombok @Builder with Inheritance: www.baeldung.com/lombok-buil…
-
Lombok’s @Builder annotation and inheritance: reinhard.codes/2015/09/16/…
-
Lombok:讓 JAVA 代碼更優雅:blog.didispace.com/java-lombok…
-
Lombok 簡介:mp.weixin.qq.com/s?__biz=MzI…