乐字节Java8核心特性之Optional类
大家好啊,上次小乐给大家介绍了Java8最最重要的一个特性——Stream流,点击可以回顾哦。
Optional<T>类(java.util.Optional)是一个容器类,代表一个值存在或不存在,原来用null表示一个值不存在,现在Optional可以更好的表达这个概念。并且可以避免空指针异常。
1、Optinal对象构建&值获取方法
| 方法 | 操作描述 |
|---|---|
| <T> Optional<T> of(T value) | 为非null的值创建一个Optional,如果值为null,则会抛出NullPointerException |
| <T> Optional<T> ofNullable(T value) | 为指定的值创建一个Optional,如果指定的值为null,则返回一个空的Optional。 |
| T get() | 如果Optional有值则将其返回,否则抛出NoSuchElementException |
| T orElse(T other) | 如果有值则将其返回,否则返回指定的其它值。 |
| T orElseGet(Supplier<? extends T> other) | 接受Supplier接口的实现用来生成默认值 |
| <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) | 如果有值则将其返回,否则抛出supplier接口创建的异常。 |
实例代码如下
Optional<String> optional = Optional.of("java8");
// NullPointerException空指针异常 值不能为空
optional = Optional.of(null);
optional = Optional.ofNullable("java8");
System.out.println(optional.get());
System.out.println(optional.orElse("java"));
System.out.println(optional.orElseGet(()-> "java"));
System.out.println(optional.orElseThrow(()->new RuntimeException()));
// 值可空 推荐使用
optional = Optional.ofNullable(null);
// 运行时抛出NoSuchElementException异常
System.out.println(optional.get());
System.out.println(optional.orElse("java"));
System.out.println(optional.orElseGet(()-> "java"));
System.out.println(optional.orElseThrow(()->new RuntimeException()));
2、Optional 逻辑判断操作
这里可以使用Optional提供的API相关方法来执行逻辑判断操作 .
| 方法 | 操作描述 |
|---|---|
| boolean isPresent() | 如果Optional实例值存在返回true,否则返回false |
| void ifPresent(Consumer<? super T> consumer) | 如果Optional实例有值则为其调用consumer,否则不做处理 。隐士地其中进行了null判断。 |
| Optional<U> map(Function<? super T, ? extends U> mapper) | 如果有值,则对其执行调用map参数中的函数得到返回值,否则返回空Optional |
| T orElseGet(Supplier<? extends T> other) | 接受Supplier接口的实现用来生成默认值 |
| <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) | 如果有值则将其返回,否则抛出supplier接口创建的异常。 |
3、用户记录查询-消除null判断
以用户模块为例,UserService中提供queryUserById方法供客户端调用,如下:
public User queryUserById(Integer userId){
return null;
}
客户端调用Java8以前逻辑代码为例避免null通常为如下形式
User user= userService.queryUserById(10);
if(null != user){
System.out.println("匹配到该用户");
/**
* 执行其他操作
*/
}else{
System.out.println("用户不存在");
}
使用Optional 形式如下:
Optional<User> userOptional = Optional.ofNullable(user);
// 使用isPresent 方法进行判断
if(userOptional.isPresent()){
System.out.println("匹配到该用户");
/**
* 执行其他操作
*/
}else{
System.out.println("用户不存在");
}
当然,既然使用了Optional了,对于if else 的代码通常也是可以给省略掉 如下(程序逻辑只关注非空的情况,使用ifPresent 进行if判断):
// 使用ifPresent 执行if 判断操作
userOptional.ifPresent((u)->{
System.out.println("匹配到该用户");
/**
* 执行其他操作
*/
});
使用map orElse方法同样也可以执行if else的逻辑判断 如下:
userOptional.map((u) -> {
System.out.println("匹配到该用户!");
/**
* 执行其他操作
*/
return u;
}).orElse(null);
4、业务逻辑结果返回
通常,在在web网站时,对于用户登录逻辑进行处理时,如果登录的结果通常由后端接口来返回,而对于后端代码的实现在Java8以前可能是这样的 借助ResultInfo 返回用户登录结果
ResultInfo resultInfo=null;
user = userService.queryUserByUnameAndUpwd(uname,upwd);
// 判断user对象是否为空
if (user == null) { // 用户不存在
resultInfo=new ResultInfo(300,"登录失败,用户名或密码错误",null);
}else{
resultInfo=new ResultInfo(200,"登录成功",user);
}
此时,使用Optional map orElse同样可以进行替换如下(省去if else 的逻辑代码)
userOptional= Optional.ofNullable(userService.queryUserByUnameAndUpwd(uname,upwd));
// 执行 map orElse 判断
userOptional.map((u)->{
/**
* 如果用户记录存在
*/
return new ResultInfo(200,"登录成功",u);
}).orElse(new ResultInfo(300,"登录失败,用户名或密码错误",null));
5、Optional 值过滤
| 方法 | 描述 |
|---|---|
| Optional<T> filter(Predicate<? super T> predicate) | 如果有值并且满足断言条件返回包含该值的Optional,否则返回空Optional |
filter方法在Optinal对象非空的基础之上加入断言判断操作,方法返回值仍然为Optinal。这种操作对于参数判断提供很大便利,例如参数满足指定条件的后续操作sql查询操作字符串拼接,常见的servlet处理多请求页面转发处理等操作。
页面转发过滤
在使用Servlet 编写web 后端代码对页面进行转发控制时,通常会通过一个Servelt 来转发不同步请求以及相应页面,使用Optional 来达到参数合法性过滤的操作,相关如下:
String action = "main";
Optional<String> optional = Optional.ofNullable(action);
String page = optional.filter(a -> a.trim().equals("") || a.equals("main"))
.map((str) -> {
System.out.println("转发到主页面");
return "main";
}).orElseGet(() -> {
System.out.println("抓发到首页面");
return "index";
});
多条件动态sql参数过滤
对于web应用多条件查询操作,通常后端在进行处理时更多的是对sql进行动态拼接来达到最终目的(搜索引擎除外),此时Optional同样能够用于该场景下参数值过滤操作,相关代码如下:
String title = "";
String type = "";
StringBuilder sqlBuilder = new StringBuilder("select * from blog where user_id=2019 ");
Optional.ofNullable(title)
.filter(StringUtils::isNoneBlank)
.ifPresent(t -> {
sqlBuilder.append(" and title " + "%" + title + "%");
});
Optional.ofNullable(type)
.filter(t -> !t.trim().equals("") && !t.equals("1"))
.ifPresent(t -> {
sqlBuilder.append(" and type = " + type);
});
6、Optional应用
Java8中Optional的引入,使得开发避免了大量Null指针的出现,借助相关方法避免了if else这种繁琐的逻辑代码编写,对于其应用,在处理空的场景下应用较多,对于if else的逻辑场景,同样使用Optional让程序更加简洁,同时使用Optional可以实现代码的链式处理。
乐字节Java8核心特性之Optional类的更多相关文章
- 乐字节-Java8核心特性实战之Lambda表达式
大家好,小乐又来给大家分享Java8核心特性了,上一篇文章是<乐字节|Java8核心实战-接口默认方法>,这次就来讲Java8核心特征之Lambda表达式. Java8 引入Lambda表 ...
- 乐字节-Java8核心特性实战之Stream(流)
说起流,我们会想起手机 ,电脑组装流水线,物流仓库商品包装流水线等等.如果把手机 ,电脑,包裹看做最终结果的话,那么加工商品前的各种零部件就可以看做数据源,而中间一系列的加工作业操作,就可以看做流的处 ...
- 乐字节Java8核心特性之方法引用
大家好,我是乐字节的小乐,上一次我们说到了Java8核心特性之函数式接口,接下来我们继续了解Java8又一核心特性--方法引用. Java8 中引入方法引用新特性,用于简化应用对象方法的调用, 方法引 ...
- 乐字节-Java8核心特性实战之函数式接口
什么时候可以使用Lambda?通常Lambda表达式是用在函数式接口上使用的.从Java8开始引入了函数式接口,其说明比较简单:函数式接口(Functional Interface)就是一个有且仅有一 ...
- 乐字节-Java8核心特性实战-接口默认方法
JAVA8已经发布很久,是自java5(2004年发布)之后Oracle发布的最重要的一个版本.其中包括语言.编译器.库.工具和JVM等诸多方面的新特性,对于国内外互联网公司来说,Java8是以后技术 ...
- 乐字节-Java8新特性之Optional
上一篇小乐带大家了解了Java新特性之Stream,接下来将会继续述说Java新特性之Optional Optional<T>类(java.util.Optional)是一个容器类,代表一 ...
- 乐字节-Java8新特性-接口默认方法
总概 JAVA8 已经发布很久,而且毫无疑问,java8是自java5(2004年发布)之后的最重要的版本.其中包括语言.编译器.库.工具和JVM等诸多方面的新特性. Java8 新特性列表如下: 接 ...
- 乐字节-Java8新特性之Base64和重复注解与类型注解
上一篇小乐给大家说了<乐字节-Java8新特性之Date API>,接下来小乐继续给大家说一说Java8新特性之Base64和重复注解与类型注解. 一.Base64 在Java 8中,内置 ...
- 乐字节-Java8新特性之Date API
上一篇文章,小乐给大家带来了Java8新特性之Optional,接下来本文将会给大家介绍Java8新特性之Date API 前言: Java 8通过发布新的Date-Time API来进一步加强对日期 ...
随机推荐
- Oracle学习笔记_03_单行函数
1.SQL函数: 单行函数.多行函数 单行函数: 操作数据对象 接受参数返回一个结果 只对一行进行变换 每行返回一个结果 可以转换数据类型 ...
- javaScript-基础篇(一)
1.如何插入JS 使用<script>标签在HTML网页中插入JavaScript代码.注意, <script>标签要成对出现,并把JavaScript代码写在<scri ...
- 2018.6.21 HOLTEK HT49R70A-1 Source Code analysis
Cange note: “Reading TMR1H will latch the contents of TMR1H and TMR1L counter to the destination”? F ...
- codeforces 652D D. Nested Segments(离散化+sort+树状数组)
题目链接: D. Nested Segments time limit per test 2 seconds memory limit per test 256 megabytes input sta ...
- 多线程编程-pthread 未定义的引用
多线程编程时用到函数库 pthread.h ,但是该函数库不是linux默认的函数库,所以编译c文件时,需要在目标代码文件后加上 -lpthread参数. 1.未加上 -lpthread 编译时,报错 ...
- STL defalloc.h
defalloc.h . // Filename: defalloc.h . . // Comment By: 凝霜 . // E-mail: mdl2009@vip.qq.com . // Blog ...
- 标准模板库(STL)学习指南之List链表
本文转载自天极网,原文地址:http://www.yesky.com/255/1910755.shtml.转载请注明 什么是STL呢?STL就是Standard Template Library,标准 ...
- [转]HTTP中cache-control的应用及说明
网页的缓存是由http消息头中的“Cache-control”来控制的,常见的取值有private.no-cache.max-age.must-revalidate等,默认为private.其作用根据 ...
- bzoj 3998 弦论 —— 后缀自动机
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3998 关于相同子串算一个还是算多个,其实就是看一种状态的 right 集合是否加上 Pare ...
- 问题11:如何进行反向迭代 & 如何实现反向迭代
# 有关列表问题,参考:Python:列表list 案例: 需求:实现一个连续浮点数发生器FloatRange(和range类似),根据给定范围(start,end)和步进值(step),产生一系列连 ...