@Builder

......鲍勃是你的叔叔:用于创建对象的无懈可击的花式裤子!
@Builder 在lombok v0.12.0中作为实验特征介绍。
@Builder获得了@Singular支持,并lombok从lombok v1.16.0 升级到主程序包。
@Builder@Singular增加,因为龙目岛v1.16.8一个明确的方法。
@Builder.Default 功能已在lombok v1.16.16中添加。

Overview

@Builder标注生产络合剂的API为你的类。

@Builder 允许您使用以下代码自动生成使您的类可实例化所需的代码:
Person.builder().name("Adam Savage").city("San Francisco").job("Mythbusters").job("Unchained Reaction").build();

@Builder可以放在类,构造函数或方法上。虽然“在类上”和“在构造函数上”模式是最常见的用例,但@Builder最容易用“方法”用例来解释。

@Builder(从现在开始调用目标)注释的方法会导致生成以下7件事:

  • 一个名为的内部静态类*Foo*Builder,具有与静态方法相同的类型参数(称为构建器)。
  • 构建器中目标的每个参数的一个私有非静态非最终字段。
  • 构建器中:一个包私有no-args空构造函数。
  • 构建器中:对于目标的每个参数,类似于“setter”的方法:它具有与该参数相同的类型和相同的名称。它返回构建器本身,以便可以链接setter调用,如上例所示。
  • 构建器中build()调用方法的方法,传入每个字段。它返回与目标返回的相同类型。
  • 构建器中:一个明智的toString()实现。
  • 在包含目标的类中:一种builder()方法,它创建构建器的新实例。

如果该元素已经存在,则将以静默方式跳过每个列出的生成元素(忽略参数计数并仅查看名称)。这包括构建器本身:如果该类已经存在,则lombok将简单地开始在此已存在的类中注入字段和方法,除非当然要存在要注入的字段/方法。您可能不会在构建器类上放置任何其他方法(或构造函数)生成lombok注释; 例如,您不能放置@EqualsAndHashCode构建器类。

@Builder可以为集合参数/字段生成所谓的“奇异”方法。它们采用1个元素而不是整个列表,并将该元素添加到列表中。例如:Person.builder().job("Mythbusters").job("Unchained Reaction").build();将导致该List<String> jobs字段中包含2个字符串。要获得此行为,需要使用注释字段/参数@Singular。该功能有自己的文档。

既然“方法”模式已经清楚了,那么@Builder在构造函数上添加注释的功能类似; 实际上,构造函数只是具有特殊语法来调用它们的静态方法:它们的“返回类型”是它们构造的类,它们的类型参数与类本身的类型参数相同。

最后,应用于@Builder类就好像您已添加@AllArgsConstructor(access = AccessLevel.PACKAGE)到类中并将@Builder注释应用于此all-args构造函数。这仅适用于您自己没有编写任何显式构造函数的情况。如果您确实有一个显式构造函数,请将@Builder注释放在构造函数而不是类上。

如果使用@Builder生成构建器来生成自己的类的实例(除非添加@Builder到不返回自己类型的方法,否则总是如此),您可以使用@Builder(toBuilder = true)在类中生成实例方法调用toBuilder(); 它会创建一个以该实例的所有值开头的新构建器。您可以将@Builder.ObtainVia注释放在参数(如果是构造函数或方法)或字段(如果@Builder是类型)上,以指示从该实例获取该字段/参数的值的替代方法。例如,您可以指定要调用的方法:@Builder.ObtainVia(method = "calculateFoo")

构建器类的名称是*Foobar*Builder,其中Foobar目标返回类型的简化,标题框形式- 即@Builder构造函数和类型的类型名称,以及@Builderon方法的返回类型的名称。例如,如果@Builder应用于名为的类com.yoyodyne.FancyList<T>,则构建器名称将为FancyListBuilder<T>。如果@Builder应用于返回的方法,void则将命名构建器VoidBuilder

构建器的可配置方面是:

  • 生成器的类名(默认:返回类型+“生成器”)
  • 版本()方法的名称(默认:"build"
  • 生成器()方法的名称(默认:"builder"
  • 如果你想toBuilder()(默认:否)

所有选项均从其默认值更改的示例用法:
@Builder(builderClassName = "HelloWorldBuilder", buildMethodName = "execute", builderMethodName = "helloWorld", toBuilder = true)

@Builder.Default

如果在构建会话期间从未设置某个字段/参数,则它始终为0 / null/ false。如果您已经放置@Builder了一个类(而不是方法或构造函数),则可以直接在该字段上指定默认值,并使用以下内容对该字段进行注释@Builder.Default
@Builder.Default private final long created = System.currentTimeMillis();

@Singular

通过使用注释注释其中一个参数(如果使用方法或构造函数进行注释@Builder)或字段(如果使用注释类@Builder@Singular,lombok将该构建器节点视为集合,并生成2个“加法器”方法而不是“ setter'方法。一个向集合添加单个元素,另一个将另一个集合的所有元素添加到集合中。将不生成仅设置集合(替换已添加的任何内容)的setter。还生成了“清晰”方法。这些“单一”构建器非常复杂,以保证以下属性:

  • 调用时build(),生成的集合将是不可变的。
  • 在调用之后调用“adder”方法之一或“clear”方法build()不会修改任何已生成的对象,并且如果build()稍后再次调用,则会生成自生成构建器以来添加了所有元素的另一个集合。
  • 生成的集合将被压缩到最小的可行格式,同时保持高效。

@Singular只能应用于lombok已知的集合类型。目前,支持的类型是:

  • java.util

    • IterableCollectionListArrayList在一般情况下由压缩的不可修改的支持)。
    • SetSortedSetNavigableSet(由一个聪明的大小不可修改HashSetTreeSet在一般情况下支持)。
    • MapSortedMapNavigableMap(由一个聪明的大小不可修改HashMapTreeMap在一般情况下支持)。
  • 番石榴的com.google.common.collect
    • ImmutableCollectionImmutableList(由构建器功能支持ImmutableList)。
    • ImmutableSetImmutableSortedSet(由这些类型的构建器功能支持)。
    • ImmutableMapImmutableBiMapImmutableSortedMap(由这些类型的构建器功能支持)。
    • ImmutableTable(由构建器功能支持ImmutableTable)。

如果您的标识符是用通用英语编写的,则lombok假定其上的任何集合的名称@Singular是英语复数,并将尝试自动单独化该名称。如果可以,add-one方法将使用此名称。例如,如果调用了您的集合statuses,则会自动调用add-one方法status。您还可以通过将单数形式作为参数传递给注释来明确指定标识符的单数形式,如下所示:@Singular("axis") List<Line> axes;
如果lombok无法单独标识您的标识符,或者它不明确,则lombok将生成错误并强制您明确指定单数名称。

下面的代码段没有显示lombok为@Singular字段/参数生成的内容,因为它相当复杂。您可以在此处查看代码段。

With Jackson

您可以自定义构建器类的自定义部分,例如向构建器类添加另一个方法,或者在构建器类中注释方法。Lombok将生成您不手动添加的所有内容,并将其放入此构建器类中。例如,如果您尝试将jackson配置为对集合使用特定子类型,则可以编写如下内容:

@Value @Builder
@JsonDeserialize(builder = JacksonExample.JacksonExampleBuilder.class)
public class JacksonExample {@Singular private List<Foo> foos;@JsonPOJOBuilder(withPrefix = "")public static class JacksonExampleBuilder implements JacksonExampleBuilderMeta {}private interface JacksonExampleBuilderMeta {@JsonDeserialize(contentAs = FooImpl.class) JacksonExampleBuilder foos(List<? extends Foo> foos)}
}

With Lombok

import lombok.Builder;
import lombok.Singular;
import java.util.Set;@Builder
public class BuilderExample {@Builder.Default private long created = System.currentTimeMillis();private String name;private int age;@Singular private Set<String> occupations;
}

Vanilla Java

import java.util.Set;public class BuilderExample {private long created;private String name;private int age;private Set<String> occupations;BuilderExample(String name, int age, Set<String> occupations) {this.name = name;this.age = age;this.occupations = occupations;}private static long $default$created() {return System.currentTimeMillis();}public static BuilderExampleBuilder builder() {return new BuilderExampleBuilder();}public static class BuilderExampleBuilder {private long created;private boolean created$set;private String name;private int age;private java.util.ArrayList<String> occupations;BuilderExampleBuilder() {}public BuilderExampleBuilder created(long created) {this.created = created;this.created$set = true;return this;}public BuilderExampleBuilder name(String name) {this.name = name;return this;}public BuilderExampleBuilder age(int age) {this.age = age;return this;}public BuilderExampleBuilder occupation(String occupation) {if (this.occupations == null) {this.occupations = new java.util.ArrayList<String>();}this.occupations.add(occupation);return this;}public BuilderExampleBuilder occupations(Collection<? extends String> occupations) {if (this.occupations == null) {this.occupations = new java.util.ArrayList<String>();}this.occupations.addAll(occupations);return this;}public BuilderExampleBuilder clearOccupations() {if (this.occupations != null) {this.occupations.clear();}return this;}public BuilderExample build() {// complicated switch statement to produce a compact properly sized immutable set omitted.Set<String> occupations = ...;return new BuilderExample(created$set ? created : BuilderExample.$default$created(), name, age, occupations);}@java.lang.Overridepublic String toString() {return "BuilderExample.BuilderExampleBuilder(created = " + this.created + ", name = " + this.name + ", age = " + this.age + ", occupations = " + this.occupations + ")";}}
}

Supported configuration keys:

lombok.builder.flagUsage = [warning | error] (default: not set)
lombok.singular.useGuava = [true | false] (default: false)
lombok.singular.auto = [true | false] (default: true)

Small print

@Singular支持java.util.NavigableMap/Set仅在您使用JDK1.8或更高版本进行编译时才有效。

您无法手动提供@Singular节点的部分或全部部分; Lombok生成的代码太复杂了。如果要手动控制(部分)与某个字段或参数关联的构建器代码,请不要@Singular手动使用和添加所需的所有内容。

排序集合(java.util中:SortedSetNavigableSetSortedMapNavigableMap,番石榴:ImmutableSortedSetImmutableSortedMap)要求该集合的类型参数有自然顺序(实现java.util.Comparable)。无法Comparator在构建器中传递显式内容。

如果目标集合来自包, An ArrayList用于将添加的元素存储为@Singular标记字段的调用方法java.util即使集合是集合或映射也是如此。由于lombok确保生成的集合被压缩,因此无论如何都必须构建集合或映射的新后备实例,并且ArrayList在构建过程中将数据存储为将其存储为映射或集合更有效。此行为不是外部可见的,是当前实现java.util配方的实现细节@Singular @Builder

随着toBuilder = true应用到方法,注解的方法的任何类型的参数本身也必须在返回类型出现。

@Builder.Default删除字段 上的初始化程序并将其存储在静态方法中,以确保在构建中指定值时,根本不会执行此初始化程序。这是否意味着初始化不能引用thissuper或任何非静态成员。如果lombok为您生成构造函数,它还将使用初始化程序初始化此字段。

各种众所周知的关于nullity的注释会导致插入空检查,并将其复制到构建器的'setter'方法的参数中。有关详细信息,请参阅Getter / Setter文档的小字体。

原文:https://www.jianshu.com/p/5e42ecede166

lombok @Builder注解的使用相关推荐

  1. Java中lombok @Builder注解使用详解

    简介 Lombok大家都知道,在使用POJO过程中,它给我们带来了很多便利,省下大量写get.set方法.构造器.equal.toString方法的时间.除此之外,通过@Builder注解,lombo ...

  2. lombok 下的@Builder注解用法

    转载:https://blog.csdn.net/qq_35568099/article/details/80438538 pom依赖 <dependency><groupId> ...

  3. lombok中的builder注解居然是一种设计模式:让我们了解一下超级实用的“建造者模式”吧

    lombok中的builder注解本质上是为你生成了一个构造器Builder类,通过这个类我们可以构造出带此注解的对象.本质上它实现了设计模式中一种经典的模式:建造者模式 1.认识: ①一句话来说:封 ...

  4. 关于lombok中的Builder注解解析

    文章最前: 我是Octopus,这个名字来源于我的中文名--章鱼:我热爱编程.热爱算法.热爱开源.所有源码在我的个人github :这博客是记录我学习的点点滴滴,如果您对 Python.Java.AI ...

  5. lombok 基础注解之 @Builder

    最全的 lombok 注解详情(随着版本不定时更新) 一.注解介绍 @Builder 注解为类生成相对略微复杂的构建器 API 它作用于类,将其变成建造者模式 可以以链的形式调用 初始化实例对象生成的 ...

  6. 如何给Lombok Builder提供默认值

    来源:生活点亮技术 1.概览 在这个教程中,我们将研究如何基于Lombok在实现 Builder模式时为属性提供默认值. 请务必阅读这篇Lombok简介 . 2.Maven依赖 在本教程中,我们将使用 ...

  7. lombok @Builder 是如何实现的

    lombok 可能大家都用过,有一个 @Builder 注解可以很方便的在对象初始化的时候填充属性值,不用大量的调用 setter 方法,代码更易于阅读与编写.不知道大家有没有想过 @Builder ...

  8. 20190905 Lombok常用注解

    Lombok常用注解 val 用于声明类型,将从初始化表达式推断出类型,仅适用于局部变量和foreach循环,而不适用于字段.声明的局部变量为final变量. Java自带类型推断随着JDK版本提升越 ...

  9. 构建器builder模式 + lombok @Builder的介绍及使用

    参考 https://juejin.im/post/6844903862600466439 注意⚠️: 带有@Builder的类,如果加了@NoArgsConstructor,那么也必须加@AllAr ...

最新文章

  1. RelativeLayout(相对布局)的分析
  2. pxe安装linux后命令不可用,CentOS7下的PXE无人值守系统安装(亲测成功)
  3. 计算机组成原理——关于数据对齐存储
  4. mybatis学习(35):sql-where
  5. Centos7-Mysql-5.6.41一主两从的搭建
  6. memcached 快速入门
  7. 学习笔记之《Android应用案例开发大全》(全部调试过代码)
  8. VB.net绘制tan函数图像
  9. python与数据思维基础笔记第一章_python学习笔记-第1章节 基础知识
  10. python量化策略——混合择时策略(动量效应+pe_ttm、pb估值+美林时钟)——股债轮动
  11. c:一个长方体表面积体积的计算
  12. python怎么进入虚拟环境_Python 中如何使用 virtualenv 管理虚拟环境
  13. Educational Codeforces Round 81 (Rated for Div. 2)
  14. Mac提示来自身份不明的开发者
  15. vcs中一个simv的诞生过程
  16. 安卓 GPS定位程序APP 闪退问题
  17. 前端工作时必备网站(纯手工总结)
  18. 银河系计算机运算,9个月漫长运算 计算机首次模拟银河系
  19. C# 发送xml报文到用友U8生成凭证系列二(基础代码)
  20. VS误删sln项目文件怎么办

热门文章

  1. 【机器学习基础】关于Scikit-Learn,你不一定知道的10件事
  2. 【实战】用机器学习来提升你的用户增长(二)
  3. 推荐系统里,可以用蒸馏吗?
  4. 阅读应用的社交还能做什么?
  5. CVTE(WEB后台开发)
  6. Windows XP \Windows 2003启动过程的学习及故障分析处理(四)
  7. HTML5和CSS3不仅仅是两项新的Web技术标准
  8. 深度学习入门篇--手把手教你用 TensorFlow 训练模型
  9. ora-04021 无法锁表的解决办法
  10. Shell之/bin/bash脚本的基础实战