如何给Lombok Builder提供默认值

640?wx_fmt=gif640?wx_fmt=jpeg

来源:生活点亮技术

1.概览

在这个教程中,我们将研究如何基于Lombok在实现 Builder模式时为属性提供默认值。

请务必阅读这篇Lombok简介 。

2.Maven依赖

在本教程中,我们将使用Lombok ,因此,需要添加一个Maven依赖:


 
  1. <dependency>

  2.    <groupId>org.projectlombok</groupId>

  3.    <artifactId>lombok</artifactId>

  4.    <version>1.16.18</version>

  5.    <scope>provided</scope>

  6. </dependency>

3.基于Lombok Builder的POJO

首先,让我们看看 Lombok如何帮助我们从实现 Builder模式所需的样板代码中解脱出来。 我们将从一个简单的 POJO开始:


 
  1. public class Pojo {

  2.    private String name;

  3.    private boolean original;

  4. }

为了使这个类可用,我们需要给每个字段实现一个getter。另外,如果希望将这个类用于ORM,我们可能需要一个默认构造函数。

除了这些,我们还需要一个用于构建这个POJOBuilder类。有了Lombok,我们就可以通过一些简单的注解来实现上面这样功能了:


 
  1. @Getter

  2. @Builder

  3. @NoArgsConstructor

  4. @AllArgsConstructor

  5. public class Pojo {

  6.    private String name;

  7.    private boolean original;

  8. }

4.期望的效果

让我们以单元测试的形式来定义想要达到的效果。

最重要,也是最基本的要求是使用builder方法构建对象之后要有默认值:


 
  1. @Test

  2. public void givenBuilderWithDefaultValue_ThenDefaultValueIsPresent() {

  3.    Pojo build = Pojo.builder()

  4.        .build();

  5.    Assert.assertEquals("foo", build.getName());

  6.    Assert.assertTrue(build.isOriginal());

  7. }

当然,这个测试用例肯定会失败,因为@Builder注解并不会给属性赋默认值。 稍后,我们会让测试通过。 如果使用了依赖默认构造函数的ORM框架,那么我们应该先从默认构造函数开始:


 
  1. @Test

  2. public void givenBuilderWithDefaultValue_NoArgsWorksAlso() {

  3.    Pojo build = Pojo.builder()

  4.        .build();

  5.    Pojo pojo = new Pojo();

  6.    Assert.assertEquals(build.getName(), pojo.getName());

  7.    Assert.assertTrue(build.isOriginal() == pojo.isOriginal());

  8. }

在这个阶段,这个测试用例通过了,因为都没有默认值。 现在,让我们看看如何让这两个测试用例都通过!

5.LombokBuilder.Default注解

自从Lombokv1.16.16之后,我们就可以使用@Builder.Default注解:


 
  1. // 添加在类上的注解同上

  2. public class Pojo {

  3.    @Builder.Default

  4.    private String name = "foo";

  5.    @Builder.Default

  6.    private boolean original = true;

  7. }

这个注解简单易读,但也有一些缺陷。

有了这个注解,默认值将将与构造函数一起出现,那么第一个测试用例将通过。不幸的是,由于@NoArgsConstructor不会得到默认值,因此第二个测试用例失败了。即使无参构造函数不是 Lombok自动生成而是显式编写的,也取不到默认值。

Builder.Default注解的这种副作用从一开始就有,可能还会持续很长时间。

6.初始化Builder

为了让这两个测试用例通过,我们可以尝试通过在一个极简的Builder中定义默认值:


 
  1. // 添加在类上的注解同上

  2. public class Pojo {

  3.    private String name = "foo";

  4.    private boolean original = true;


  5.    public static class PojoBuilder {

  6.        private String name = "foo";

  7.        private boolean original = true;

  8.    }

  9. }

通过这种方式,这两个测试用例都通过了。 很不幸,代价是代码重复。对于具有数十个属性的POJO,维护双重初始化可能会出错。

但是,如果愿意付出这个代价,我们还应该注意一件事。如果我们使用IDE中的重构功能来重命名类,静态内部类将不会自动重命名。这样,Lombok就找不到它了,我们的代码就出错了。

为了消除这种风险,我们可以再配置一下Builder注解:


 
  1. // 添加在类上的注解同上

  2. @Builder(builderClassName = "PojoBuilder")

  3. public class Pojo {

  4.    private String name = "foo";

  5.    private boolean original = true;


  6.    public static class PojoBuilder {

  7.        private String name = "foo";

  8.        private boolean original = true;

  9.    }

  10. }

7.使用toBuilder参数

@Builder还支持在原始类的实例中生成一个Builder实例。默认情况下这个特性是关闭的。我们可以通过在Builder注解中配置toBuilder参数来启用:


 
  1. // class annotations as before

  2. // 添加在类上的注解同上

  3. @Builder(toBuilder = true)

  4. public class Pojo {

  5.    private String name = "foo";

  6.    private boolean original = true;

  7. }

这样,我们就可以避免*双重初始化 *。

当然,这也是有代价的。我们必须通过 实例化这个类来创建一个Builder实例。因此,相关的测试用例也必须修改:


 
  1. @Test

  2. public void givenBuilderWithDefaultValue_ThenDefaultValueIsPresent() {

  3.    Pojo build =  new Pojo().toBuilder()

  4.        .build();

  5.    Assert.assertEquals("foo", build.getName());

  6.    Assert.assertTrue(build.isOriginal());

  7. }


  8. @Test

  9. public void givenBuilderWithDefaultValue_thenNoArgsWorksAlso() {

  10.    Pojo build = new Pojo().toBuilder()

  11.        .build();

  12.    Pojo pojo = new Pojo();

  13.    Assert.assertEquals(build.getName(), pojo.getName());

  14.    Assert.assertTrue(build.isOriginal() == pojo.isOriginal());

  15. }

同样,这两个测试用例会全部通过。因此使用无参构造函数与使用Builder具有相同的默认值。

8.总结

至此,我们已经展示了为Lombok Builder提供默认值的几种方法。

Builder.Default注解的副作用也很明显。但是,其他几个方案也有缺点。所以我们必须结合自身情况谨慎选择。

照例,文中用到的代码都可以在 GitHub上找到。

原文链接:https://www.baeldung.com/lombok-builder-default-value

作者: baeldung

译者: helloworldtang


·END·


 近期热文:

640?wx_fmt=png

看完,赶紧点个“好看”鸭

点鸭点鸭

↓↓↓↓

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值