⼀种简化源码提⾼编程效率的⼯具,⽤于⽣成常⽤的代码。
两个包:
lombok
lombok.experimental (实验的特性)
org.projectlombok
lombok
${lombok.version}
@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插件。