Lombok是一款非常实用Java工具,可用来帮助开发人员消除Java的冗长代码,尤其是对于简单的Java对象(POJO)。
前言
博猪日常最多的也就是CRUD,对于这个网上争议最多的插件来说,网上说的一些争议点没有遇到过,当然归根结底的原因基本上仅仅用到了getter/setter方法,同时也让我的对象显得简洁而已,toString和equals没什么用到。
另外说一个很重要的原因就是,博猪感觉就是技术是时代的产物,既然存在,那必然也有它存在的道理。所以我并不认为网上说的都对,也不全错,做技术嘛,要有个辩论思维,不要人云亦云嘛。
工作原理
Lombok 是一个开源的项目,它提供了一系列的代码生成工具,用于简化 Java 代码的开发。
Lombok 的工作原理是通过使用 Java 的反射机制,在编译时生成代码,从而实现自动生成 Setter 和 Getter 方法、toString、equals、hashCode 等方法。
具体来说,Lombok 使用了 Java 的注解来标识需要生成的代码,然后在编译时通过反射机制解析这些注解,生成相应的代码。
例如,使用 Lombok 的 @Getter 和 @Setter 注解可以生成 Setter 和 Getter 方法,使用 Lombok 的 @ToString 注解可以生成 toString 方法,使用 Lombok 的 @EqualsAndHashCode 注解可以生成 equals 和 hashCode 方法。
Lombok 还提供了一些其他的功能,例如生成构造函数、getters 和 setters 的组合、链式设置等。这些功能都可以通过使用 Lombok 的注解来实现。
Lombok 的工作原理是基于 Java 的反射机制,在编译时生成代码,从而简化 Java 代码的开发。
Lombok 安装
IDEA搜索Lombok插件
提示 另外需要注意的是,在使用lombok注解的时候记得要导入lombok.jar包到工程,如果使用的是Maven的工程项目的话,要在其pom.xml中添加依赖如下:
1 2 3 4 5 6 7
| <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version>1.18.12</version> <scope>provided</scope> </dependency>
|
Lombok注解说明
val
:用在局部变量前面,相当于将变量声明为final@NonNull
:给方法参数增加这个注解会自动在方法内对该参数进行是否为空的校验,如果为空,则抛出NPE
(NullPointerException)@Cleanup
:自动管理资源,用在局部变量之前,在当前变量范围内即将执行完毕退出之前会自动清理资源,自动生成try-finally
这样的代码来关闭流@Getter/@Setter
:用在属性上,再也不用自己手写setter
和getter
方法了,还可以指定访问范围@ToString
:用在类上,可以自动覆写toString
方法,当然还可以加其他参数,例如@ToString(exclude=”id”)
排除id属性,或者@ToString(callSuper=true, includeFieldNames=true)
调用父类的toString
方法,包含所有属性@EqualsAndHashCode
:用在类上,自动生成equals
方法和hashCode
方法@NoArgsConstructor
, @RequiredArgsConstructor
and @AllArgsConstructor
:用在类上,自动生成无参构造和使用所有参数的构造函数以及把所有+ @NonNull属性作为参数的构造函数,如果指定
staticName = “of”`参数,同时还会生成一个返回类对象的静态工厂方法,比使用构造函数方便很多@Data
:注解在类上,相当于同时使用了@ToString
、@EqualsAndHashCode
、@Getter
、@Setter
和@RequiredArgsConstrutor
这些注解,对于POJO类十分有用@Value
:用在类上,是@Data
的不可变形式,相当于为属性添加final声明,只提供getter方法,而不提供setter方法@Builder
:用在类、构造器、方法上,为你提供复杂的builder APIs,让你可以像如下方式一样调用Person.builder().name("Adam Savage").city("San Francisco").job("Mythbusters").job("Unchained Reaction").build()
;更多说明参考Builder@SneakyThrows
:自动抛受检异常,而无需显式在方法上使用throws语句@Synchronized
:用在方法上,将方法声明为同步的,并自动加锁,而锁对象是一个私有的属性$lock
或$LOCK
,而java中的synchronized关键字锁对象是this,锁在this或者自己的类对象上存在副作用,就是你不能阻止非受控代码去锁this或者类对象,这可能会导致竞争条件或者其它线程错误@Getter(lazy=true)
:可以替代经典的Double Check Lock样板代码@Log
:根据不同的注解生成不同类型的log对象,但是实例名称都是log,有六种可选实现类@CommonsLog
Creates log = org.apache.commons.logging.LogFactory.getLog(LogExample.class);@Log
Creates log = java.util.logging.Logger.getLogger(LogExample.class.getName());@Log4j
Creates log = org.apache.log4j.Logger.getLogger(LogExample.class);@Log4j2
Creates log = org.apache.logging.log4j.LogManager.getLogger(LogExample.class);@Slf4j
Creates log = org.slf4j.LoggerFactory.getLogger(LogExample.class);@XSlf4j
Creates log = org.slf4j.ext.XLoggerFactory.getXLogger(LogExample.class);
Lombok代码示例
val示例
1 2 3 4 5 6 7 8 9
| public static void main(String[] args) { val sets = new HashSet<String>(); val lists = new ArrayList<String>(); val maps = new HashMap<String, String>(); final Set<String> sets2 = new HashSet<>(); final List<String> lists2 = new ArrayList<>(); final Map<String, String> maps2 = new HashMap<>(); }
|
@NonNull
示例
1 2 3 4 5 6 7 8 9 10 11
| public void notNullExample(@NonNull String string) { string.length(); }
public void notNullExample(String string) { if (string != null) { string.length(); } else { throw new NullPointerException("null"); } }
|
@Cleanup
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| public static void main(String[] args) { try { @Cleanup InputStream inputStream = new FileInputStream(args[0]); } catch (FileNotFoundException e) { e.printStackTrace(); } InputStream inputStream = null; try { inputStream = new FileInputStream(args[0]); } catch (FileNotFoundException e) { e.printStackTrace(); } finally { if (inputStream != null) { try { inputStream.close(); } catch (IOException e) { e.printStackTrace(); } } } }
|
@Getter/@Setter
示例
1 2 3 4
| @Setter(AccessLevel.PUBLIC) @Getter(AccessLevel.PROTECTED) private int id; private String shape;
|
@ToString
示例
1 2 3 4 5 6 7 8 9 10
| @ToString(exclude = "id", callSuper = true, includeFieldNames = true) public class LombokDemo { private int id; private String name; private int age; public static void main(String[] args) { System.out.println(new LombokDemo()); } }
|
@EqualsAndHashCode
示例
1 2 3 4 5
| @EqualsAndHashCode(exclude = {"id", "shape"}, callSuper = false) public class LombokDemo { private int id; private String shape; }
|
@NoArgsConstructor
, @RequiredArgsConstructor
and @AllArgsConstructor
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| @NoArgsConstructor @RequiredArgsConstructor(staticName = "of") @AllArgsConstructor public class LombokDemo { @NonNull private int id; @NonNull private String shape; private int age; public static void main(String[] args) { new LombokDemo(1, "circle"); LombokDemo.of(2, "circle"); new LombokDemo(); new LombokDemo(1, "circle", 2); } }
|
@Data
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| import lombok.Data; @Data public class Menu { private String shopId; private String skuMenuId; private String skuName; private String normalizeSkuName; private String dishMenuId; private String dishName; private String dishNum; private float thresHold = 0; private float newThresHold = 0; private float totalScore = 0; }
|
@Value
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| @Value public class LombokDemo { @NonNull private int id; @NonNull private String shap; private int age; private final int id; public int getId() { return this.id; } ... }
|
@Builder
示例
1 2 3 4 5 6 7 8 9 10 11 12 13
| @Builder public class BuilderExample { private String name; private int age; @Singular private Set<String> occupations; public static void main(String[] args) { LombokDemo3 test = LombokDemo3.builder().age(11).name("test") .occupation("1") .occupation("2") .build(); } }
|
@Singular可以为集合类型的参数或字段生成一种特殊的方法, 它采用修改列表中一个元素而不是整个列表的方式,可以是增加一个元素,也可以是删除一个元素。
在使用@Singular注释注释一个集合字段(使用@Builder注释类),lombok会将该构建器节点视为一个集合,并生成两个adder方法而不是setter方法。
生成代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
| public LombokDemo3.LombokDemo3Builder occupation(String occupation) { if (this.occupations == null) { this.occupations = new ArrayList(); }
this.occupations.add(occupation); return this; }
public LombokDemo3.LombokDemo3Builder occupations(Collection<? extends String> occupations) { if (occupations == null) { throw new NullPointerException("occupations cannot be null"); } else { if (this.occupations == null) { this.occupations = new ArrayList(); }
this.occupations.addAll(occupations); return this; } }
public LombokDemo3.LombokDemo3Builder clearOccupations() { if (this.occupations != null) { this.occupations.clear(); }
return this; }
|
Builder.Default
1 2 3 4 5 6 7 8 9 10 11 12 13
| @Builder @ToString public class BuilderDefaultExample {
@Builder.Default private final String id = UUID.randomUUID().toString(); private String username;
@Builder.Default private long insertTime = System.currentTimeMillis();
}
|
@SneakyThrows
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| import lombok.SneakyThrows; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.InputStream; import java.io.UnsupportedEncodingException; public class Test { @SneakyThrows() public void read() { InputStream inputStream = new FileInputStream(""); } @SneakyThrows public void write() { throw new UnsupportedEncodingException(); } public void read() throws FileNotFoundException { InputStream inputStream = new FileInputStream(""); } public void write() throws UnsupportedEncodingException { throw new UnsupportedEncodingException(); } }
|
@Synchronized
示例
1 2 3 4 5 6 7 8 9 10 11 12 13
| public class SynchronizedDemo { @Synchronized public static void hello() { System.out.println("world"); } private static final Object $LOCK = new Object[0]; public static void hello() { synchronized ($LOCK) { System.out.println("world"); } } }
|
@Getter(lazy = true)
示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
| public class GetterLazyExample { @Getter(lazy = true) private final double[] cached = expensive(); private double[] expensive() { double[] result = new double[1000000]; for (int i = 0; i < result.length; i++) { result[i] = Math.asin(i); } return result; } }
import java.util.concurrent.atomic.AtomicReference; public class GetterLazyExample { private final AtomicReference<java.lang.Object> cached = new AtomicReference<>(); public double[] getCached() { java.lang.Object value = this.cached.get(); if (value == null) { synchronized (this.cached) { value = this.cached.get(); if (value == null) { final double[] actualValue = expensive(); value = actualValue == null ? this.cached : actualValue; this.cached.set(value); } } } return (double[]) (value == this.cached ? null : value); } private double[] expensive() { double[] result = new double[1000000]; for (int i = 0; i < result.length; i++) { result[i] = Math.asin(i); } return result; } }
|