学习下Lombok。

关于POJO

Java面向对象编程中的特性中有封闭性和安全性。封闭性即对类中的域变量进行封闭操作,即用private来修饰他们。如此一来,其他类就不能对该变量访问了。这样,我们就将这些变量封闭在了类的内部,提高了数据的安全性。

当我们想要操作这些域变量的时候,有两种办法。第一种是通过public方式的构造器(或称构造函数),对象一实例化就对该变量赋值。第二种就是通过set和get方法对变量进行赋值和取值。这样就能提高域变量的安全性,同时又保证了域变量的封装型。

所以当我们创建POJO类时,都会毫不犹豫的让开发工具对域变量生成set、get方法。虽然不是我们自己手动添加(快捷键或菜单快速生成),但每个类都要做重复的生成操作也是一件很烦人的事。而且当变量名或者是修饰符改变了的话,我们就要删除set、get方法并重新生成,又是一项重复又枯燥的操作。Lombok就是一个为了提高生产效率,让我们免去这些重复操作的神器。

Lombok的介绍

Project Lombok makes java a spicier language by adding 'handlers' that know how to build and compile simple, boilerplate-free, not-quite-java code.

上面这段话是官方的介绍,意思是Lombok通过一些特殊的处理,可以让Java编程变得简洁和快速。

Lombok的使用

Lombok能以简单的注解形式来简化Java代码的编写,提高开发人员的开发效率。比如在编译时自动为属性生成构造器、getter/setter、equals、hashcode、toString方法。意思就是说,在源码中不需要编写getter和setter方法,但是在编译生成的字节码文件中却会有getter和setter方法。这样就省去了手动重建这些代码的麻烦,使代码看起来更简洁些。

使用Lombok需要引用Jar包依赖,而在Maven中添加依赖十分简单。

<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.16.20</version>
<scope>provided</scope>
</dependency>

下面是Lombok提供的一些常用注解的介绍与使用。

@Data注解

@Data注解使用在类上,会为类的所有属性自动生成setter/getter、equals、canEqual、hashCode、toString方法。要注意的是,如果是final修饰符修饰的属性,则不会为该属性生成setter方法。

@Data
public class DataExample {
private final String name; @Setter(AccessLevel.PACKAGE)
private int age; private double score; private String[] tags; @ToString(includeFieldNames=true)
@Data(staticConstructor="of")
public static class Exercise<T> {
private final String name;
private final T value;
}
}

在开发中建议尽量少直接使用@Data注解,而是换成@Setter、@Getter、@NoArgsConstructor、@AllArgsConstructor、@ToString。

@Getter/@Setter注解

因为@Data集合了@ToString、@EqualsAndHashCode、@Getter/@Setter、@RequiredArgsConstructor的所有特性,很多时候我们可能并不需要那么多的特性,因此Lombok提供了更精细的注解@Getter/@Setter,这个注解使用在属性上会为该属性自动生成getter/setter方法。

public class GetterSetterExample {
@Getter
@Setter
private int age = 10; @Setter(AccessLevel.PROTECTED)
private String name; @Override
public String toString() {
return String.format("%s (age: %d)", name, age);
}
}

@NonNull注解

@NonNull注解使用在属性或构造器上,Lombok会生成一个非空的声明,可用于校验参数,能帮助避免空指针。

public class NonNullExample extends Something {
private String name; public NonNullExample(@NonNull Person person) {
super("Hello");
this.name = person.getName();
}
}

如果不使用@NonNull注解,则要手动对属性/构造器进行非空校验。

public class NonNullExample extends Something {
private String name; public NonNullExample(Person person) {
super("Hello");
if (person == null) {
throw new NullPointerException("person");
}
this.name = person.getName();
}
}

@Cleanup注解

@Cleanup注解能帮助我们自动调用close()方法,很大程度上简化了代码。

public class CleanupExample {
public static void main(String[] args) throws IOException {
@Cleanup
InputStream in = new FileInputStream(args[0]); @Cleanup
OutputStream out = new FileOutputStream(args[1]); byte[] b = new byte[10000];
while (true) {
int r = in.read(b);
if (r == -1) break;
out.write(b, 0, r);
}
}
}

如果不使用@Lombok注解,则需要手动调用colse()关闭输入/输出流。

public class CleanupExample {
public static void main(String[] args) throws IOException {
InputStream in = new FileInputStream(args[0]);
try {
OutputStream out = new FileOutputStream(args[1]);
try {
byte[] b = new byte[10000];
while (true) {
int r = in.read(b);
if (r == -1) break;
out.write(b, 0, r);
}
} finally {
if (out != null) {
out.close();
}
}
} finally {
if (in != null) {
in.close();
}
}
}
}

@EqualsAndHashCode注解

默认情况下,会使用所有非静态(non-static)和非瞬态(non-transient)属性来生成equals和hasCode,也能通过exclude注解来排除一些属性。

@EqualsAndHashCode(exclude={"id", "shape"})
public class EqualsAndHashCodeExample {
private transient int transientVar = 10;
private String name;
private double score;
private Shape shape = new Square(5, 10);
private String[] tags;
private int id; public String getName() {
return this.name;
} @EqualsAndHashCode(callSuper=true)
public static class Square extends Shape {
private final int width, height;
public Square(int width, int height) {
this.width = width;
this.height = height;
}
}
}

@ToString注解

在类上使用@ToString注解,Lombok会生成一个toString()方法,默认情况下,会输出类名、所有属性(会按照属性定义顺序),用逗号来分割。

而通过将includeFieldNames参数设为true,就能明确的输出toString()属性。

@ToString(exclude="id")
public class ToStringExample {
private static final int STATIC_VAR = 10;
private String name;
private Shape shape = new Square(5, 10);
private String[] tags;
private int id; public String getName() {
return this.getName();
} @ToString(callSuper=true, includeFieldNames=true)
public static class Square extends Shape {
private final int width, height; public Square(int width, int height) {
this.width = width;
this.height = height;
}
}
}

@NoArgsConstructor、@RequiredArgsConstructor和@AllArgsConstructor注解

三个注解分别对应无参构造器、部分参数构造器和全参构造器。Lombok没法实现多种参数构造器的重载。

@RequiredArgsConstructor(staticName = "of")
@AllArgsConstructor(access = AccessLevel.PROTECTED)
public class ConstructorExample<T> {
private int x, y; @NonNull
private T description; @NoArgsConstructor
public static class NoArgsExample {
@NonNull
private String field;
}
}

加上staticName参数会生成一个of()的静态方法。

默认生成的方法事public的,如果想要生成其他方法修饰符修饰的方法,可以设置access参数。

Lombok简单分析

Lombok的基本原理是在编译的时候通过解析注解去自动生成相应的代码。JDK5在引入注解的同时,提供了两种解析注解的方式,一种是运行时解析,一种是编译时解析。

运行时编译

运行时能够解析的注解,必须是将注解定义中的@Retention注解设置为RUNTIME,这样才可以通过Java的反射机制拿到使用注解的类信息。在java.lang.reflect反射包中提供了一个AnnotatedElement接口,这个接口定义了获取注解信息的几个方法,Class、Constructor、Field、Method和Package等都实现了这个接口。

编译时解析

编译时的解析有两种机制:

1.Annotation Processing Tool(apt)

apt自JDK5产生,在JDK7已经被标记为过期,不推荐使用,且在JDK8中已经被彻底删除了。自JDK6开始可以使用Pluggable Annotation Processing API来替换它。apt被替换的原因主要有两个,一个是它的api都在com.sun.mirror非标准包下,一个是它没有被集成到javac中,需要额外运行。

2.Pluggable Annotation Processing API(JSR 269)

JSR 269自JDK6加入,作为apt的替代方案,它解决了apt的两个问题,javac在执行的时候会调用实现了该API的程序,这样我们可以对编译器做一些增强。

事实上,Lombok本质上就是一个实现了JSR 269 API的程序。

Lombok的插件

Lombok的插件是提供给IDE的,目的是为了方便开发者在编写源码的时候能看到Lombok通过注解自动生成的代码,没有安装插件的情况下也可能会发生编译错误。

Idea安装Lombok插件

Idea可以到插件库(Plugins)中搜索Lombok Plugin直接安装,简单又方便。

Eclipse安装Lombok插件

Eclipes则需要到官网下载插件到本地,再通过命令行安装插件。

1.到官网下载Lombok插件:https://projectlombok.org/download.html

2.命令行切换到Lombok插件的下载目录,运行命令:java -jar lombok.jar。

3.在弹出的可视化界面中选择Eclipse的安装/解压目录并点击Install/Update即可。

安装完成之后,Eclipse的安装/解压路径下会多出一个lombok.jar文件,并且其eclipse.ini配置文件也会添加两行Lombok配置的代码:

-javaagent:lombok.jar
-Xbootclasspath/a:lombok.jar

这样,在添加完Lombok提供的注解之后就可以在Outline中查看到Lombok自动生成的代码了。

总结

Lombok是个好东西,能有效提高开发效率,使代码变得简洁,维护起来也变得简单了。但是有一个大的缺点就是Lombok不支持多个参数的构造器重载。

另外要注意的是,手动写的代码会覆盖Lombok自动生成的代码。比如如果在源代码中手写了getter/setter方法的话,会覆盖Lombok生成的getter/setter方法,或者说Lombok在编译的时候就不会去生成已经存在的getter/setter。

"如果必须完成一件自己不喜欢的事,最好的做法就是尽可能快地将这件事做好,然后无后续地结束。"

lombok的介绍、使用、简单分析和插件的更多相关文章

  1. 插件lombok的介绍安装

    Lombok插件 介绍一个不错的Eclipse插件Lombok 该插件对Log4j简化的代码,因为不大,所以jar包也存在呢! Lombox是Eclipse的一个插件,用来自动生成Java代码,减少手 ...

  2. CSipSimple 简单分析

    简介 CSipSimple是一款可以在android手机上使用的支持sip的网络电话软件,可以在上面设置使用callda网络电话.连接使用方式最好是使用wifi,或者3g这样上网速度快,打起电话来效果 ...

  3. 接口测试从未如此简单 - Postman (Chrome插件)

    接口测试从未如此简单 - Postman (Chrome插件) 一个非常有力的Http Client工具用来测试Web服务的, 我这里来介绍如何用它测试restful web service 注:转载 ...

  4. 简单分析Java的HashMap.entrySet()的实现

    关于Java的HashMap.entrySet(),文档是这样描述的:这个方法返回一个Set,这个Set是HashMap的视图,对Map的操作会在Set上反映出来,反过来也是.原文是 Returns ...

  5. WebRTC介绍及简单应用

    WebRTC介绍及简单应用 WebRTC,即Web Real-Time Communication,web实时通信技术.简单地说就是在web浏览器里面引入实时通信,包括音视频通话等. WebRTC实时 ...

  6. FFmpeg源码简单分析:libswscale的sws_scale()

    ===================================================== FFmpeg的库函数源码分析文章列表: [架构图] FFmpeg源码结构图 - 解码 FFm ...

  7. AbstractQueuedSynchronizer的简单分析

    说明:本作者是文章的原创作者,转载请注明出处:本文地址:http://www.cnblogs.com/qm-article/p/7955781.html 一.AbstractQueuedSynchro ...

  8. FFmpeg的HEVC解码器源代码简单分析:CTU解码(CTU Decode)部分-TU

    ===================================================== HEVC源代码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpe ...

  9. FFmpeg的HEVC解码器源代码简单分析:解码器主干部分

    ===================================================== HEVC源代码分析文章列表: [解码 -libavcodec HEVC 解码器] FFmpe ...

随机推荐

  1. SQLServer 高效 分页存储过程

    /********************************************************************** 参数:@PrimaryKey 主键,@OrderBy 排 ...

  2. 尝试在iOS上使用gRPC

    简介 gRPC,google的远程过程调用框架,传输协议使用 HTTP2, 序列化协议使用 protobuf.gRPC 使用 HTTP2 传输协议传输 protobuf 序列化的二进制数据,有极高的效 ...

  3. rem与em的使用和区别详解【转】

    目录 最大的问题是 主要区别  rem 单位如何转换为像素值  em 单位如何转换为像素值 Em 单位的遗传效果 Em 继承的例子 浏览器设置 HTML 元素字体大小的影响 没有设置 HTML 字体大 ...

  4. day09 作业

    简述定义函数的三种方式 空函数.无参函数.有参函数 简述函数的返回值 函数内部代码经过一系列的逻辑处理返回的结果 函数没有返回值,默认返回None 函数可以通过return返回出返回值 return可 ...

  5. 网络唤醒(WOL)全解指南:原理篇【转】

    转自:https://blog.csdn.net/z5859095/article/details/82819075 什么是网络唤醒网络唤醒(Wake-on-LAN,WOL)是一种计算机局域网唤醒技术 ...

  6. 3-14 Pandas绘图

      1.魔法指令:%matplotlib inline :数据画图 In [1]: %matplotlib inline import pandas as pd In [2]: import nump ...

  7. ansible 模块 roles

    setup 作用,用来查看用看内部的详细信息 ansible_all_ipv4_addresses # ipv4的所有地址 ansible_all_ipv6_addresses # ipv6的所有地址 ...

  8. JS高阶---继承模式(原型链继承)

    [前言] 之前已经介绍了对象创建的五种模式,下面看下继承模式 本节介绍下<原型链继承> [主体] 验证如下: 关键点: .

  9. 2015年第六届蓝桥杯C/C++程序设计本科B组决赛 ——居民集会(编程大题)

    标题:居民集会 蓝桥村的居民都生活在一条公路的边上,公路的长度为L,每户家庭的 位置都用这户家庭到公路的起点的距离来计算,第i户家庭距起点的距 离为di.每年,蓝桥村都要举行一次集会.今年,由于村里的 ...

  10. Leetcode148-Sort_List

    Sort_List 在LeetCode 里面,因为只有归并排序的时间复杂度为O(1),所以快速排序用不了,前面两个都没用直接看最后一个归并排序. 冒泡排序(超时了) public ListNode s ...