以下文章來源于Java中文社群 裁僧,作者老王
世界上只有一種英雄主義,就是看清生活的真相之后依然熱愛生活慕购。
對于 Lombok 我相信大部分人都不陌生,但對于它的實現(xiàn)原理以及缺點卻鮮為人知茬底,而本文將會從 Lombok 的原理出發(fā)沪悲,手擼一個簡易版的 Lombok,讓你理解這個熱門技術(shù)背后的執(zhí)行原理阱表,以及它的優(yōu)缺點殿如。
簡介
在講原理之前,我們先來復習一下 Lombok (老司機可直接跳過本段)最爬。
Lombok 是一個非常熱門的開源項目 (https://github.com/rzwitserloot/lombok)涉馁,使用它可以有效的解決 Java 工程中那些繁瑣又重復代碼,例如 Setter爱致、Getter烤送、toString、equals糠悯、hashCode 以及非空判斷等帮坚,都可以使用 Lombok 有效的解決。
使用
1.添加 Lombok 插件
在 IDE 中必須安裝 Lombok 插件互艾,才能正常調(diào)用被 Lombok 修飾的代碼试和,以 Idea 為例,添加的步驟如下:
- 點擊 File > Settings > Plugins 進入插件管理頁面
- 點擊 Browse repositories...
- 搜索 Lombok Plugin
- 點擊 Install plugin 安裝插件
- 重啟 IntelliJ IDEA
2.添加 Lombok 庫
接下來我們需要在項目中添加最新的 Lombok 庫阅悍,如果是 Maven 項目,直接在 pom.xml 中添加如下配置:
<!-- https://mvnrepository.com/artifact/org.projectlombok/lombok -->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.12</version>
<scope>provided</scope>
</dependency>
</dependencies>
如果是 JDK 9+ 可使用模塊的方式添加,配置如下:
<path>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.12</version>
</path>
</annotationProcessorPaths>
3.使用 Lombok
接下來到了前半部分中最重要的 Lombok 使用環(huán)節(jié)了节视,我們先來看在沒有使用 Lombok 之前的代碼:
public class Person {
private Integer id;
private String name;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
這是使用 Lombok 之后的代碼:
@Getter
@Setter
public class Person {
private Integer id;
private String name;
}
可以看出在 Lombok 之后晦墙,用一個注解就搞定了之前所有 Getter/Setter 的代碼,讓代碼瞬間優(yōu)雅了很多肴茄。
Lombok 所有注解如下:
val
:用在局部變量前面晌畅,相當于將變量聲明為 final;@NonNull
:給方法參數(shù)增加這個注解會自動在方法內(nèi)對該參數(shù)進行是否為空的校驗寡痰,如果為空抗楔,則拋出 NPE(NullPointerException);@Cleanup
:自動管理資源拦坠,用在局部變量之前连躏,在當前變量范圍內(nèi)即將執(zhí)行完畢退出之前會自動清理資源,自動生成 try-finally 這樣的代碼來關(guān)閉流贞滨;@Getter/@Setter
:用在屬性上入热,再也不用自己手寫 setter 和 getter 方法了,還可以指定訪問范圍晓铆;@ToString
:用在類上可以自動覆寫 toString 方法勺良,當然還可以加其他參數(shù),例如 @ToString(exclude=”id”) 排除 id 屬性骄噪,或者 @ToString(callSuper=true, includeFieldNames=true) 調(diào)用父類的 toString 方法尚困,包含所有屬性;@EqualsAndHashCode
:用在類上自動生成 equals 方法和 hashCode 方法链蕊;@NoArgsConstructor, @RequiredArgsConstructor and @AllArgsConstructor
:用在類上事甜,自動生成無參構(gòu)造和使用所有參數(shù)的構(gòu)造函數(shù)以及把所有 @NonNull 屬性作為參數(shù)的構(gòu)造函數(shù),如果指定 staticName="of" 參數(shù)滔韵,同時還會生成一個返回類對象的靜態(tài)工廠方法逻谦,比使用構(gòu)造函數(shù)方便很多;@Data
:注解在類上陪蜻,相當于同時使用了 @ToString邦马、@EqualsAndHashCode、@Getter囱皿、@Setter 和 @RequiredArgsConstrutor 這些注解勇婴,對于 POJO 類十分有用;@Value
:用在類上嘱腥,是 @Data 的不可變形式耕渴,相當于為屬性添加 final 聲明,只提供 getter 方法齿兔,而不提供 setter 方法橱脸;@Builder
:用在類础米、構(gòu)造器、方法上添诉,為你提供復雜的 builder APIs屁桑,讓你可以像如下方式一樣調(diào)用Person.builder().name("xxx").city("xxx").build();@SneakyThrows
:自動拋受檢異常栏赴,而無需顯式在方法上使用 throws 語句蘑斧;@Synchronized
:用在方法上,將方法聲明為同步的须眷,并自動加鎖竖瘾,而鎖對象是一個私有的屬性lock或者LOCK,而 Java 中的 synchronized 關(guān)鍵字鎖對象是 this花颗,鎖在 this 或者自己的類對象上存在副作用捕传,就是你不能阻止非受控代碼去鎖 this 或者類對象,這可能會導致競爭條件或者其它線程錯誤扩劝;@Getter(lazy=true)
:可以替代經(jīng)典的 Double Check Lock 樣板代碼庸论;@Log
:根據(jù)不同的注解生成不同類型的 log 對象,但是實例名稱都是 log棒呛,有六種可選實現(xiàn)類@CommonsLog
Creates log = org.apache.commons.logging.LogFactory.getLog(LogExample.class);@Log
Creates log = java.util.logging.Logger.getLogger(LogExample.class.getName());@Log4j
Creates log = org.apache.log4j.Logger.getLogger(LogExample.class);@Log4j2
Creates log = org.apache.logging.log4j.LogManager.getLogger(LogExample.class);@Slf4j
Creates log = org.slf4j.LoggerFactory.getLogger(LogExample.class);@XSlf4j
Creates log = org.slf4j.ext.XLoggerFactory.getXLogger(LogExample.class);
它們的具體使用如下:
① val 使用
val sets = new HashSet<String>();
// 相當于
final Set<String> sets = new HashSet<>();
② NonNull 使用
public void notNullExample(@NonNull String string) {
string.length();
}
// 相當于
public void notNullExample(String string) {
if (string != null) {
string.length();
} else {
throw new NullPointerException("null");
}
}
③ Cleanup 使用
public static void main(String[] args) {
try {
@Cleanup InputStream inputStream = new FileInputStream(args[0]);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
// 相當于
InputStream inputStream = null;
try {
inputStream = new FileInputStream(args[0]);
} catch (FileNotFoundException e) {
e.printStackTrace();
} finally {
if (inputStream != null) {
try {
inputStream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
④ Getter/Setter 使用
@Setter(AccessLevel.PUBLIC)
@Getter(AccessLevel.PROTECTED)
private int id;
private String shap;
⑤ ToString 使用
@ToString(exclude = "id", callSuper = true, includeFieldNames = true)
public class LombokDemo {
private int id;
private String name;
private int age;
public static void main(String[] args) {
// 輸出 LombokDemo(super=LombokDemo@48524010, name=null, age=0)
System.out.println(new LombokDemo());
}
}
⑥ EqualsAndHashCode 使用
@EqualsAndHashCode(exclude = {"id", "shape"}, callSuper = false)
public class LombokDemo {
private int id;
private String shap;
}
⑦ NoArgsConstructor聂示、RequiredArgsConstructor、AllArgsConstructor 使用
@NoArgsConstructor
@RequiredArgsConstructor(staticName = "of")
@AllArgsConstructor
public class LombokDemo {
@NonNull
private int id;
@NonNull
private String shap;
private int age;
public static void main(String[] args) {
new LombokDemo(1, "Java");
// 使用靜態(tài)工廠方法
LombokDemo.of(2, "Java");
// 無參構(gòu)造
new LombokDemo();
// 包含所有參數(shù)
new LombokDemo(1, "Java", 2);
}
}
⑧ Builder 使用
@Builder
public class BuilderExample {
private String name;
private int age;
@Singular
private Set<String> occupations;
public static void main(String[] args) {
BuilderExample test = BuilderExample.builder().age(11).name("Java").build();
}
}
⑨ SneakyThrows 使用
public class ThrowsTest {
@SneakyThrows()
public void read() {
InputStream inputStream = new FileInputStream("");
}
@SneakyThrows
public void write() {
throw new UnsupportedEncodingException();
}
// 相當于
public void read() throws FileNotFoundException {
InputStream inputStream = new FileInputStream("");
}
public void write() throws UnsupportedEncodingException {
throw new UnsupportedEncodingException();
}
}
⑩ Synchronized 使用
public class SynchronizedDemo {
@Synchronized
public static void hello() {
System.out.println("world");
}
// 相當于
private static final Object $LOCK = new Object[0];
public static void hello() {
synchronized ($LOCK) {
System.out.println("world");
}
}
}
? Getter(lazy = true) 使用
public class GetterLazyExample {
@Getter(lazy = true)
private final double[] cached = expensive();
private double[] expensive() {
double[] result = new double[1000000];
for (int i = 0; i < result.length; i++) {
result[i] = Math.asin(i);
}
return result;
}
}
// 相當于
import java.util.concurrent.atomic.AtomicReference;
public class GetterLazyExample {
private final AtomicReference<java.lang.Object> cached = new AtomicReference<>();
public double[] getCached() {
java.lang.Object value = this.cached.get();
if (value == null) {
synchronized (this.cached) {
value = this.cached.get();
if (value == null) {
final double[] actualValue = expensive();
value = actualValue == null ? this.cached : actualValue;
this.cached.set(value);
}
}
}
return (double[]) (value == this.cached ? null : value);
}
private double[] expensive() {
double[] result = new double[1000000];
for (int i = 0; i < result.length; i++) {
result[i] = Math.asin(i);
}
return result;
}
}
原理分析
我們知道 Java 的編譯過程大致可以分為三個階段:
- 解析與填充符號表
- 注解處理
- 分析與字節(jié)碼生成
而 Lombok 正是利用「注解處理」這一步進行實現(xiàn)的条霜,Lombok 使用的是 JDK 6 實現(xiàn)的 JSR 269: Pluggable Annotation Processing API (編譯期的注解處理器) 催什,它是在編譯期時把 Lombok 的注解代碼,轉(zhuǎn)換為常規(guī)的 Java 方法而實現(xiàn)優(yōu)雅地編程的宰睡。
這一點可以在程序中得到驗證,比如本文剛開始用 @Data
實現(xiàn)的代碼:
在我們編譯之后气筋,查看 Person 類的編譯源碼發(fā)現(xiàn)拆内,代碼竟然是這樣的:
可以看出 Person 類在編譯期被注解翻譯器修改成了常規(guī)的 Java 方法,添加 Getter宠默、Setter麸恍、equals、hashCode 等方法搀矫。
Lombok 的執(zhí)行流程如下:
可以看出抹沪,在編譯期階段,當 Java 源碼被抽象成語法樹 (AST) 之后瓤球,Lombok 會根據(jù)自己的注解處理器動態(tài)的修改 AST融欧,增加新的代碼 (節(jié)點),在這一切執(zhí)行之后卦羡,再通過分析生成了最終的字節(jié)碼 (.class) 文件噪馏,這就是 Lombok 的執(zhí)行原理麦到。
手擼一個 Lombok
我們實現(xiàn)一個簡易版的 Lombok 自定義一個 Getter 方法,我們的實現(xiàn)步驟是:
- 自定義一個注解標簽接口欠肾,并實現(xiàn)一個自定義的注解處理器瓶颠;
- 利用 tools.jar 的 javac api 處理 AST (抽象語法樹)
- 使用自定義的注解處理器編譯代碼。
這樣就可以實現(xiàn)一個簡易版的 Lombok 了刺桃。
1.定義自定義注解和注解處理器
首先創(chuàng)建一個 MyGetter.java
自定義一個注解粹淋,代碼如下:
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Retention(RetentionPolicy.SOURCE) // 注解只在源碼中保留
@Target(ElementType.TYPE) // 用于修飾類
public @interface MyGetter { // 定義 Getter
}
再實現(xiàn)一個自定義的注解處理器,代碼如下:
import com.sun.source.tree.Tree;
import com.sun.tools.javac.api.JavacTrees;
import com.sun.tools.javac.code.Flags;
import com.sun.tools.javac.code.Type;
import com.sun.tools.javac.processing.JavacProcessingEnvironment;
import com.sun.tools.javac.tree.JCTree;
import com.sun.tools.javac.tree.TreeMaker;
import com.sun.tools.javac.tree.TreeTranslator;
import com.sun.tools.javac.util.*;
import javax.annotation.processing.*;
import javax.lang.model.SourceVersion;
import javax.lang.model.element.Element;
import javax.lang.model.element.TypeElement;
import javax.tools.Diagnostic;
import java.util.Set;
@SupportedSourceVersion(SourceVersion.RELEASE_8)
@SupportedAnnotationTypes("com.example.lombok.MyGetter")
public class MyGetterProcessor extends AbstractProcessor {
private Messager messager; // 編譯時期輸入日志的
private JavacTrees javacTrees; // 提供了待處理的抽象語法樹
private TreeMaker treeMaker; // 封裝了創(chuàng)建AST節(jié)點的一些方法
private Names names; // 提供了創(chuàng)建標識符的方法
@Override
public synchronized void init(ProcessingEnvironment processingEnv) {
super.init(processingEnv);
this.messager = processingEnv.getMessager();
this.javacTrees = JavacTrees.instance(processingEnv);
Context context = ((JavacProcessingEnvironment) processingEnv).getContext();
this.treeMaker = TreeMaker.instance(context);
this.names = Names.instance(context);
}
@Override
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
Set<? extends Element> elementsAnnotatedWith = roundEnv.getElementsAnnotatedWith(MyGetter.class);
elementsAnnotatedWith.forEach(e -> {
JCTree tree = javacTrees.getTree(e);
tree.accept(new TreeTranslator() {
@Override
public void visitClassDef(JCTree.JCClassDecl jcClassDecl) {
List<JCTree.JCVariableDecl> jcVariableDeclList = List.nil();
// 在抽象樹中找出所有的變量
for (JCTree jcTree : jcClassDecl.defs) {
if (jcTree.getKind().equals(Tree.Kind.VARIABLE)) {
JCTree.JCVariableDecl jcVariableDecl = (JCTree.JCVariableDecl) jcTree;
jcVariableDeclList = jcVariableDeclList.append(jcVariableDecl);
}
}
// 對于變量進行生成方法的操作
jcVariableDeclList.forEach(jcVariableDecl -> {
messager.printMessage(Diagnostic.Kind.NOTE, jcVariableDecl.getName() + " has been processed");
jcClassDecl.defs = jcClassDecl.defs.prepend(makeGetterMethodDecl(jcVariableDecl));
});
super.visitClassDef(jcClassDecl);
}
});
});
return true;
}
private JCTree.JCMethodDecl makeGetterMethodDecl(JCTree.JCVariableDecl jcVariableDecl) {
ListBuffer<JCTree.JCStatement> statements = new ListBuffer<>();
// 生成表達式 例如 this.a = a;
JCTree.JCExpressionStatement aThis = makeAssignment(treeMaker.Select(treeMaker.Ident(
names.fromString("this")), jcVariableDecl.getName()), treeMaker.Ident(jcVariableDecl.getName()));
statements.append(aThis);
JCTree.JCBlock block = treeMaker.Block(0, statements.toList());
// 生成入?yún)? JCTree.JCVariableDecl param = treeMaker.VarDef(treeMaker.Modifiers(Flags.PARAMETER),
jcVariableDecl.getName(), jcVariableDecl.vartype, null);
List<JCTree.JCVariableDecl> parameters = List.of(param);
// 生成返回對象
JCTree.JCExpression methodType = treeMaker.Type(new Type.JCVoidType());
return treeMaker.MethodDef(treeMaker.Modifiers(Flags.PUBLIC),
getNewMethodName(jcVariableDecl.getName()), methodType, List.nil(),
parameters, List.nil(), block, null);
}
private Name getNewMethodName(Name name) {
String s = name.toString();
return names.fromString("get" + s.substring(0, 1).toUpperCase() + s.substring(1, name.length()));
}
private JCTree.JCExpressionStatement makeAssignment(JCTree.JCExpression lhs, JCTree.JCExpression rhs) {
return treeMaker.Exec(
treeMaker.Assign(
lhs,
rhs
)
);
}
}
自定義的注解處理器是我們實現(xiàn)簡易版的 Lombok 的重中之重瑟慈,我們需要繼承 AbstractProcessor
類桃移,重寫它的 init() 和 process() 方法,在 process() 方法中我們先查詢所有的變量封豪,在給變量添加對應的方法谴轮。我們使用 TreeMaker 對象和 Names 來處理 AST,如上代碼所示吹埠。
當這些代碼寫好之后第步,我們就可以新增一個 Person 類來試一下我們自定義的 @MyGetter
功能了,代碼如下:
@MyGetter
public class Person {
private String name;
}
2.使用自定義的注解處理器編譯代碼
上面的所有流程執(zhí)行完成之后缘琅,我們就可以編譯代碼測試效果了粘都。首先,我們先進入代碼的根目錄刷袍,執(zhí)行以下三條命令翩隧。
進入的根目錄如下:
① 使用 tools.jar 編譯自定義的注解器
javac -cp $JAVA_HOME/lib/tools.jar MyGetter* -d .
注意:命令最后面有一個“.”表示當前文件夾。
② 使用自定義注解器呻纹,編譯 Person 類
javac -processor com.example.lombok.MyGetterProcessor Person.java
③ 查看 Person 源碼
javap -p Person.class
源碼文件如下:
可以看到我們自定義的 getName() 方法已經(jīng)成功生成了堆生,到這里簡易版的 Lombok 就大功告成了。
Lombok 優(yōu)缺點
Lombok 的優(yōu)點很明顯雷酪,它可以讓我們寫更少的代碼吴旋,節(jié)約了開發(fā)時間奏候,并且讓代碼看起來更優(yōu)雅扮休,它的缺點有以下幾個唇撬。
缺點1:降低了可調(diào)試性
Lombok 會幫我們自動生成很多代碼,但這些代碼是在編譯期生成的吩跋,因此在開發(fā)和調(diào)試階段這些代碼可能是“丟失的”寞射,這就給調(diào)試代碼帶來了很大的不便。
缺點2:可能會有兼容性問題
Lombok 對于代碼有很強的侵入性锌钮,加上現(xiàn)在 JDK 版本升級比較快桥温,每半年發(fā)布一個版本,而 Lombok 又屬于第三方項目轧粟,并且由開源團隊維護策治,因此就沒有辦法保證版本的兼容性和迭代的速度脓魏,進而可能會產(chǎn)生版本不兼容的情況。
缺點3:可能會坑到隊友
尤其對于組人來的新人可能影響更大通惫,假如這個之前沒用過 Lombok茂翔,當他把代碼拉下來之后,因為沒有安裝 Lombok 的插件履腋,在編譯項目時珊燎,就會提示找不到方法等錯誤信息,導致項目編譯失敗遵湖,進而影響了團結(jié)成員之間的協(xié)作悔政。
缺點4:破壞了封裝性
面向?qū)ο蠓庋b的定義是:通過訪問權(quán)限控制,隱藏內(nèi)部數(shù)據(jù)延旧,外部僅能通過類提供的有限的接口訪問和修改內(nèi)部數(shù)據(jù)谋国。
也就是說,我們不應該無腦的使用 Lombok 對外暴露所有字段的 Getter/Setter 方法迁沫,因為有些字段在某些情況下是不允許直接修改的芦瘾,比如購物車中的商品數(shù)量,它直接影響了購物詳情和總價集畅,因此在修改的時候應該提供統(tǒng)一的方法近弟,進行關(guān)聯(lián)修改,而不是給每個字段添加訪問和修改的方法挺智。
小結(jié)
本文我們介紹了 Lombok 的使用以及執(zhí)行原理祷愉,它是通過 JDK 6 實現(xiàn)的 JSR 269: Pluggable Annotation Processing API (編譯期的注解處理器) ,在編譯期時把 Lombok 的注解轉(zhuǎn)換為 Java 的常規(guī)方法的赦颇,我們可以通過繼承 AbstractProcessor 類二鳄,重寫它的 init() 和 process() 方法,實現(xiàn)一個簡易版的 Lombok媒怯。但同時 Lombok 也存在這一些使用上的缺點泥从,比如:降低了可調(diào)試性、可能會有兼容性等問題沪摄,因此我們在使用時要根據(jù)自己的業(yè)務場景和實際情況,來選擇要不要使用 Lombok纱烘,以及應該如何使用 Lombok杨拐。
最后提醒一句,再好的技術(shù)也不是萬金油擂啥,就好像再好的鞋子也得適合自己的腳才行哄陶!
轉(zhuǎn)自 & 鳴謝
博主:macrozheng