⼀种简化源码提⾼编程效率的⼯具,⽤于⽣成常⽤的代码。
两个包:
lombok
lombok.experimental (实验的特性)
- <dependency>
- <groupId>org.projectlombokgroupId>
- <artifactId>lombokartifactId>
- <version>${lombok.version}version>
- dependency>

- @Data
- public class People {
- private String name;
- }
| @val | 声明变量时能自行推断变量类型, 并且自带 final 属性, |
| @Data | @ToString、@EqualsAndHashCode、@Getter、@Setter和@RequiredArgsConstrutor |
| @Slf4j | 生成 log 对象 |
| @Value | 与 @Data 类似, 有以下两点区别:
|
| @Getter | 所有属性的Getter方法 |
| @Setter | 所有属性的Setter方法 |
| @Builder | 生成链式构造器的代码 |
| @Cleanup | 安全释放或者关闭资源, 最常见的场景就是 IO 中关闭流的操作 |
| @NonNull | getter : 如果获取出来属性为 null, 则抛出 NPE setter : 如果设置属性时, 传入的值为 null, 则抛出 NPE |
| @ToString | 生成 toString() 方法 |
| @SneakyThrows | 将抛出的异常吃掉, 减少一些不必要的 try catch 代码 |
| @NoArgsConstructor | 生成一个无参构造方法 |
| @AllArgsConstructor | 添加一个包含所有属性的Constructor |
| @EqualsAndHashCode | 生成 equals() 和 hashcode() 方法 |
| @RequiredArgsConstrutor | 会生成一个包含常量,和标识了NotNull的变量的构造方法 |
被注释的部分是lombok会帮助我们自动生成的代码。
- @Slf4j
- public class Test {
- // private static final Logger log=LoggerFactory.getLogger(Test.class);
- public static void main(String[] args) {
- log.info("Hello world");
- }
- }
- Test test = Test.builder()
- .id(id)
- .page(page)
- .build();
-
- @Data
- @Builder
- public class Test {
-
- private String id;
-
- private String page;
-
- /**
- @java.beans.ConstructorProperties({"id", "page"})
- Test(Long id, int page) {
- this.id = id;
- this.page = page;
- }
- public static TestBuilder builder() {
- return new TestBuilder();
- }
- public TestBuilder toBuilder() {
- return new TestBuilder().id(this.id).page(this.page);
- }
- public static class TestBuilder {
- private Long id;
- private int page;
- TestBuilder() {}
- public TestBuilder id(Long id) {
- this.id = id;
- return this;
- }
- public TestBuilder page(int page) {
- this.page = page;
- return this;
- }
- public Test build() {
- return new Test(id, page);
- }
- public String toString() {
- return "Test.TestBuilder(id=" + this.id + ", page="
- + this.page")";
- }
- */
- }
- @Test(expected = RuntimeException.class)
- @SneakyThrows
- public void test_throw_exception() {
- when(HttpClientUtil.get(anyString()).thenThrow(new RuntimeException());
- api.test("nice");
- }
- @Data
- public class User {
-
- private Long id;
-
- private String username;
-
- private String password;
-
-
- /**
- public User() {}
- public Long getId() {return this.id;}
- public String getUsername() {return this.username;}
- public String getPassword() {return this.password;}
- public void setId(Long id) {this.id = id; }
- public void setUsername(String username) {this.username = username; }
- public void setPassword(String password) {this.password = password; }
- public boolean equals(Object o) {
- if (o == this) { return true; }
- ...
- return true;
- }
- public int hashCode() {
- final int PRIME = 59;
- int result = 1;
- final Object $id = this.getId();
- result = result * PRIME + ($id == null ? 43 : $id.hashCode());
- final Object $username = this.getUsername();
- ...
- return result;
- }
- protected boolean canEqual(Object other) {return other instanceof User;}
- public String toString() {
- return "User(id=" + this.getId() + ...+ ")";
- }
- */
- }
- }
- @Value
- public class Test {
-
- (private final) Long id;
-
- (private final) String page;
-
- /**
- @java.beans.ConstructorProperties({"id", "page"})
- public Test(Long id, String page) {
- this.id = id;
- this.page = page;
- }
- public Long getId() {return this.id;}
- public String getPage() {return this.page;}
- public boolean equals(Object o) {
- if (o == this) { return true; }
- ...
- return true;
- }
- public int hashCode() {
- final int PRIME = 59;
- int result = 1;
- final Object $id = this.getId();
- result = result * PRIME + ($id == null ? 43 : $id.hashCode());
- ...
- return result;
- }
- public String toString() {
- return "Test.TestBuilder(id=" + this.id + ", page="
- + this.page")";
- }
- */
- }
【推荐】字段偏多或构造函数参数偏多的不可变实体类推荐使用lombok来生成builder
【推荐】在Record类型数据支持之前,推荐实体类字段用lombok注解来生成setter和getter,并根据最小化原则,来决定是否只用@Getter、@Setter还是@Data
【推荐】合约式编程时,推荐使用IDEA IntelliJ > Spring > Lombok的注解的非空注解来说明参数非空
有效的减少代码量
动态。例如在增减字段时,⽆需再操⼼Getter/Setter等⽅法的修改
在⼀定程度上提⾼了代码的可读性
引⼊外部依赖,一旦在resource的包里使用了lombok,别人想看你的源码也不得不安装插件
⽣成的代码不直观,可能不按预期⽣成代码
降低了源码的完整性
属性默认值问题。如果使用 @Builder 生成代码, 属性的默认值会失效, 需要使用 @Builder.Default 标记有默认值的属性, 比如:
- @Builder
- public class UserQueryParam {
- private Long id;
- private String username;
- @Builder.Default
- private int page = 1;
- @Builder.Default
- private int size = 15;
- }
对布尔值的特殊处理
private boolean test;
1.读取属性的方法默认生成规则是is+属性名而不是get+属性名,即isTest而不是getTest
2.假如属性名自身以is开头,如isTest,则获取属性的方法依旧是isTest,而不是别扭的isIsTest
3.以上两条规则,会出现一种因为不合理导致的极端情况,即有两个属性,一个名字是isTest,另一个是test,这里本质上是设计存在问题,对于这种情况,插件的处理方式是只生成一个isTest,至于读取的是哪个属性,取决于属性的顺序,居前者优先.
@Value常用于不可变类。不可变类是指创建该类的实例后,该实例的实例变量是不可改变的。
与 @Data 类似, 主要有以下两点区别:
生成的是全参的构造器;
只有 getter 方法, 没有 setter 方法;
|
| @Value | @Data |
| @Getter | ✅ | ✅ |
| @Setter | ❌ | ✅ |
| @ToString | ✅ | ✅ |
| @EqualsAndHashCode | ✅ | ✅ |
| @RequiredArgsConstrutor | ❌ | ✅ |
| @AllArgsConstructor | ✅ | ❌ |
| @FieldDefaults(makeFinal=true, level=AccessLevel.PRIVATE) | ✅ | ❌ |
被@Value标识的注解不能与@RequestBody、jackson同时使用。
使用@Value后jackson反序列化失败而fastjson可以成功的原因:
1、jackson读取的是set方法,@Value没有生成set方法,所以失败;
2、fastjson读取的是构造函数,@Value生成了构造函数,所以成功;
扩展:使用fastjson时,使用@Builder,要检查下有没有加上@NoArgsConstructor或者@AllArgsConstructor
在编译时修改抽象语法树(AST)
- @Getter(AccessLevel.PUBLIC)
- public class User {
-
- private Long id;
-
- private String username;
-
- @Getter(AccessLevel.PRIVATE)
- private String password;
- }
- public class User {
-
- private Long id;
-
- private String username;
-
- private String password;
-
- public Long getId() {return this.id;}
-
- public String getUsername() {return this.username;
-
- // Field 优先级更高
- private String getPassword() {return this.password;}
- }
Lombok是在AST语法树环节处理,AST是一种用来描述程序代码语法结构的树形表示方式,语法树的每一个节点都代表着程序代码中的一个语法结构,例如包、类型、修饰符、运算符、接口、返回值,甚至代码注释等都可以是一个语法结构。

在Idea上,也可以通过安装插件查看代码的AST语法树,如下图所示

左边的每个属性,每个方法,都能在右侧找到对应的节点,因此通过操作AST树的节点,即可完成在编译期的代码动态添加。
自Java 6起,javac开始支持 JSR 269 Pluggable Annotation Processing API 规范,只要程序实现了该API,就能在java源码编译时调用定义的注解。Lombok的本质是依靠 JSR 269 来实现在Javac编译阶段利用“Annotation Processor”(注解处理工具)对自定义的注解进行预处理后生成真正在JVM上面执行的“Class文件”。

从上面的这个原理图上可以看出Annotation Processing是编译器在解析Java源代码和生成Class文件之间的一个步骤。
Lombok本质上就是一个实现了“JSR 269 API”的程序。在使用javac的过程中,它产生作用的具体流程如下::

javac对源代码进行分析,生成了一棵抽象语法树(AST)
运行过程中调用实现了“JSR 269 API”的Lombok程序
此时Lombok就对第一步骤得到的AST进行处理,找到@Data注解所在类对应的语法树(AST),然后修改该语法树(AST),增加getter和setter方法定义的相应树节点
javac使用修改后的抽象语法树(AST)生成字节码文件,即给class增加新的节点(代码块)
从上面的Lombok执行的流程图中可以看出,在Javac 解析成AST抽象语法树之后, Lombok 根据自己编写的注解处理器,动态地修改 AST,增加新的节点(即Lombok自定义注解所需要生成的代码),最终通过分析生成JVM可执行的字节码Class文件。
Lombok中的多个自定义注解都分别有对应的handler处理类,在Lombok中对于其自定义注解进行实际的替换、修改和处理的正是这些handler类,对于其实现的细节可以具体参考其中的代码。

使用Lombok注解省略的方法在被调用时,会报找不到定义的错误,此种情况下,需要做些特殊处理,如Intellij Idea中,需要下载安装Lombok plugin插件。
