optional java_Java Optional实践(小结)

问题描述

在大热的Spring Boot 2.0中,在将原来的泛型改为了Optional,旨在让我们的代码更简洁。

9e4e66d83facd1802e651b5226477df5.png

实践

Optional

很简单的一个类,点开它的源代码,其中所有的方法都是与null相关联的。

16f4dcd5a56f93b3804a8512dca020e2.png

这是一个简化我们处理null的类。

它就是一个容器,其中有我们想要的对象,但是该对象有时候会是空,所以我们需要使用Optional封装好的方法来获取需要的对象。从而很好地避免了空指针异常。

62c32a58cbf21d47aa52422f43898679.png

错误示范

我看到网上很多人这么写:

catRepository.findById(id).get();

下面是Spring Boot 1.5的写法,那请问:如果上面的写法是正确的,那为什么还要大费周章设计一个Optional呢?

catRepository.findOne(id);

分析

通过get是能获取到我们需要的对象。

但是看看get的源代码,这样写,抛出了NoSuchElementException异常,这个异常我们没法在全局中处理它。

public T get() {

if (value == null) {

throw new NoSuchElementException("No value present");

}

return value;

}

为什么不能再全局中处理呢?大家可以思考一下:

因为NoSuchElementException覆盖的范围太广了,只要是Optional中有null就会抛出NoSuchElementException,很多情况下都会造成这种异常,那我们究竟要给用户一个什么样的提示信息好呢?最后还是给出500服务器异常,那异常处理的意义何在呢?

所以我们需要用Optional来抛出一个有特定范围的能被全局准确处理的异常。

Cat cat = catRepository.findOne(id);

if (null == cat) {

throw new EntityNotFoundException("该实体找不到");

}

return cat;

思想都是一样,我们不过是用一种更简洁的写法实现上面的功能。

实现

没错,就像下面一样,我们只需要一行代码!

public Cat findById(Long id) {

return catRepository.findById(id).orElseThrow(EntityNotFoundException::new);

}

findById返回一个Optional,然后调用该对象的orElseThrow方法。

1e726a5fb196352468c007d00d45e275.png

orElseThrow方法,如果存在,返回包含的值,否则抛出异常。

该方法的参数是一个lamda表达式。这里就不深究lamda表达式的几种类型了,如果感兴趣可以自行研究下Function、Consumer、Predicate、Supplier这四个函数式接口的区别。

38857f1177aedab8d4e86e8fa30880bf.png

所以传一个lamda表达式进去,然后IDEA会给出警告:

b4323700d0b1e049f01c72fc23634f4a.png

Can be replaced with method reference

该lamda表达式能被一个方法引用代替,Alt + Enter,我们最终的代码就长这样:

d11abb282425ac8eb94d5a0f101b1cc3.png

这里的::是lamda表达式的一种简写,是Java8中的新特性,看着可能有点奇怪,原来,编译器比程序员聪明多了。

异常处理

@RestControllerAdvice

public class GlobalExceptionHandler {

@ExceptionHandler(EntityNotFoundException.class)

public ResponseEntity entityNotFoundHandler() {

return new ResponseEntity<>("您要找的实体不存在", HttpStatus.NOT_FOUND);

}

}

写个控制器增强,全局处理异常,这里的RestControllerAdvice又是一个组合注解:

08ffae96799d8eb675ac3f921f571c7d.png

处理异常,同时以Json的格式返回。

@Test

public void findById() throws Exception {

this.mockMvc.perform(get("/cat/1"))

.andDo(print());

}

写个控制器的单元测试,查询一个不存在的实体,运行,看控制台的打印输出:

e847a653dcabe4ff3a29d25b77872120.png

一劳永逸

一劳永逸,这是我们最喜欢的东西了。

return catRepository.findById(id).orElseThrow(EntityNotFoundException::new);

以后再查询,就这一行,再也不用去判断null了。

NotNull

正所谓条条大路通罗马,对null的一劳永逸,我们这样实现,别人也可以那样实现。

如果你在Spring的项目中打过断点调试的话,那我断定你一定见过下面这行代码:

Assert.notNull();

以下是该方法的源码,注意这里的Assert是org.springframework.util包下的:

5893727a18559e9fd2e8648922a880a0.png

刚方法用于判断null,如果为空,则抛出异常。

随便点开一个方法,都会在第一行为不该为null的参数进行判断。

bf895d968891cb4ca32060baad29408e.png

这里,不禁对整个框架肃然起敬,同样一个方法,大牛写了二十分钟,而你写了十分钟,但是你却去改了半个小时的bug。

@Nullable

可能在上面看到了我们不熟悉的注解@Nullable,表示从来没见过,这个注解干什么用的呢?

万能的StackOverflow又给出了完美的回答:

eccc9d3dc7141c2fc08b523cb4d88497.png

这会让你的代码更清晰,如果你重写这个方法,你也需要让参数可为空。通常也用于代码提示。

@Nullable和@NotNull这一对注解,没什么实际意义,只是用于代码更清晰,同时编译器能给出我们提示。

总结

之前一直抱怨Java更新的太快,学校教的是Java5之前的东西,从Java5开始有的注解,但是从来没讲过这个东西,然而去看看官方的描述:

da40f230cd97923a90c63fc9290e8a71.png

其实,Java的每次更新,都是为了我们更简洁优雅的代码而努力。去看看官方的描述,Java让我们将更多的精力放在think上,而不是code上。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值