@Builder注解使用

一、@Builder注解

资料来源: http://fendou.net.cn/index.php/a/369
https://blog.csdn.net/qq_39249094/article/details/120881578

  • 作用于类,将其变成建造者模式

  • 可以以链的形式调用

  • 初始化实例对象生成的对象是不可以变的,可以在创建对象的时候进行赋值(如果想改变的话需要在@Builder后面添加参数toBuilder=true)

  • 需要在原来的基础上修改可以加 set 方法,final 字段可以不需要初始化

  • 生成一个全参的构造函数

1.0 Lombok坐标


 org.projectlomboklombok 0.10.2

提供在设计数据实体时,对外保持private setter,而对属性的赋值采用Builder的方式,这种方式最优雅,也更符合封装的原则,不对外公开属性的写操作

@Builder声明实体,表示可以进行Builder方式初始化

@Value注解,表示只公开getter,对所有属性的setter都封闭,即private修饰,所以它不能和@Builder一起用

1.1 注解使用


@Builder@Getter@DatapublicclassUserInfo {privateStringname;privateStringemail;​@Overridepublic String toString() {return"UserInfo{"+"name='"+name+'\''+", email='"+email+'\''+'}';}​public static void main(String[] args) {UserInfo userInfo = UserInfo.builder().build();System.out.println("userInfo---->"+userInfo);​UserInfo userInfo1 = UserInfo.builder().name("zzl").email("bgood@sina.com").build();System.out.println("userInfo1---->"+userInfo1);}}

图片[1] - @Builder注解使用 - MaxSSL

1.2 注解的属性介绍


1.2.1 toBuilder

  • 设置为 true 可以对这个对象进行拷贝生成新的对象,可以再修改,默认为 false

怎么设置为true?

@Builder(toBuilder = true)

我们使用UserInfo.builder().build()创建出来之后,还可以修改对象的内容么(不使用set方法)?

图片[2] - @Builder注解使用 - MaxSSL

我们此时发现如果想对已经构建了的对象在修改的话,会出错,并找不到这个方法,我们只需要在类注解上添加@Builder(toBuilder = true)即可

@Builder(toBuilder=true)@Getterpublic class UserInfo {}

userInfo = userInfo.toBuilder().name("OK").email("zgood@sina.com").build();

1.2.2 @Builder.Default 注解

非 final 的字段可以有默认值

@Builder.Defaultprivate String name ="刘亦菲";

我们下面虽然没有对name赋值,但是输出时”name“依然会时”刘亦菲”

UserInfo userInfo = UserInfo.builder().build();System.out.println("userInfo---->"+userInfo);

final字段加不加Default都可以初始化成功,因为final字段如果第一次不是null的话,就不可修改(简单的来说,final字段有了初始值之后就不可更改)

private final Integerage=18; 

这两种写法都可以

@Builder.Defaultprivate final Integer age;

1.2.3 buildMethodName

指定创建实体类的方法名,默认值为 build

当我们指定内部静态类的方法名为“test”的时候,发现下面已经开始报错了

图片[3] - @Builder注解使用 - MaxSSL

当我们把这里改成test之后便不会报错了

图片[4] - @Builder注解使用 - MaxSSL

1.2.4 builderMethodName

指定创建内部静态类的方法名,默认值为 builder

图片[5] - @Builder注解使用 - MaxSSL

1.2.5 builderClassName

指定内部静态的类名,默认值为 “”,默认创建的类名为 thisclassBuilder

这个我不太懂,不知道怎么演示

1.2.6 access

设置 builderMethodName 的访问权限修饰符,默认为 public

共有 PUBLIC、MODULE、PROTECTED、PACKAGE、PRIVATE,其中 MODULE 是 Java 9 的新特性

access = AccessLevel.PUBLIC

1.2.7 setterPrefix

设置 setter 方法的前缀,默认为 “”

1.3 处理添加无参构造函数报错时报错


@Builder 会生成一个全参构造方法,因此就没有了无参构造方法,但当我们遇到需要无参构造方法时就会发生问题,这个时候手写或者加上 @NoArgsConstructor 都会报错

图片[6] - @Builder注解使用 - MaxSSL

1.3.1 处理方案1

加上 @AllArgsConstructor

图片[7] - @Builder注解使用 - MaxSSL

1.3.2 处理方案2

使用 @Builder 对一个 DTO 实现一个构造器,但是在做 Json 反序列化的时候发生错误,原因就是缺少无参公共的构造函数,而手动写一个无参构造函数的时候编译错误,就是和 @Builder 冲突

虽然标准的 @Builder 没法是需要私有化构造函数的,但是在某些场景下我们需要对这种标准变形,这个时候 lombok 提供了 @Tolerate 实现对冲突的兼容

使用@Tolerate注解

我们手动添加一个无参构造函数,但是当运行之后就会出现错误

图片[8] - @Builder注解使用 - MaxSSL

但是当我们在无参构造函数上添加@Tolerate注解之后就可以正常运行

图片[9] - @Builder注解使用 - MaxSSL

1.4 @Builder内部


  • 创建一个名为 ThisClassBuilder 的内部静态类,并具有和实体类相同的属性(称为构建器)

  • 在构建器中:对于目标类中的所有的属性和未初始化的 final 字段,都会在构建器中创建对应属性

  • 在构建器中:创建一个无参的 default 构造函数

  • 在构建器中:实体类中的每个参数,都会对应创建类似于 setter 的方法,方法名与该参数名相同。 并且返回值是构建器本身(便于链式调用)

  • 在构建器中:会创建一个 build 方法,调用 build 方法,就会根据设置的值进行创建实体对象

  • 在构建器中:会生成一个 toString 方法

  • 在实体类中:会创建一个 builder 方法,它的目的是用来创建构建器

@Builderpublic class User {private String username;private String password;}

public class User {private String username;private String password;​​User(Stringusername, Stringpassword) {this.username=username;this.password=password;}//在实体类中会创建一个 builder 方法,它的目的是用来创建构建器public static User.UserBuilder builder() {returnnewUser.UserBuilder();}//构建器public static class UserBuilder {//在构建器中:对于目标类中的所有的属性和未初始化的 final 字段,都会在构建器中创建对应属性private String username;private String password;//在构建器中:创建一个无参的 default 构造函数UserBuilder() {}​//在构建器中:实体类中的每个参数,都会对应创建类似于 setter 的方法,方法名与该参数名相同。 并且返回值是构建器本身(便于链式调用)public User.UserBuilde rusername(Stringusername) {this.username=username;returnthis;}​public User.UserBuilderpassword(Stringpassword) {this.password=password;returnthis;}//在构建器中:会创建一个 build 方法,调用 build 方法,就会根据设置的值进行创建实体对象publicUserbuild() {return newUser(this.username, this.password);}//在构建器中:会生成一个 toString 方法 public String toString() {return"User.UserBuilder(username="+this.username+", password="+this.password+")";}}} 

© 版权声明
THE END
喜欢就支持一下吧
点赞0 分享