乐字节Java8核心特性之Optional类

时间:2022-09-21 18:38:13

大家好啊,上次小乐给大家介绍了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类的更多相关文章

  1. 乐字节-Java8核心特性实战之Lambda表达式

    大家好,小乐又来给大家分享Java8核心特性了,上一篇文章是<乐字节|Java8核心实战-接口默认方法>,这次就来讲Java8核心特征之Lambda表达式. Java8 引入Lambda表 ...

  2. 乐字节-Java8核心特性实战之Stream(流)

    说起流,我们会想起手机 ,电脑组装流水线,物流仓库商品包装流水线等等.如果把手机 ,电脑,包裹看做最终结果的话,那么加工商品前的各种零部件就可以看做数据源,而中间一系列的加工作业操作,就可以看做流的处 ...

  3. 乐字节Java8核心特性之方法引用

    大家好,我是乐字节的小乐,上一次我们说到了Java8核心特性之函数式接口,接下来我们继续了解Java8又一核心特性--方法引用. Java8 中引入方法引用新特性,用于简化应用对象方法的调用, 方法引 ...

  4. 乐字节-Java8核心特性实战之函数式接口

    什么时候可以使用Lambda?通常Lambda表达式是用在函数式接口上使用的.从Java8开始引入了函数式接口,其说明比较简单:函数式接口(Functional Interface)就是一个有且仅有一 ...

  5. 乐字节-Java8核心特性实战-接口默认方法

    JAVA8已经发布很久,是自java5(2004年发布)之后Oracle发布的最重要的一个版本.其中包括语言.编译器.库.工具和JVM等诸多方面的新特性,对于国内外互联网公司来说,Java8是以后技术 ...

  6. 乐字节-Java8新特性之Optional

    上一篇小乐带大家了解了Java新特性之Stream,接下来将会继续述说Java新特性之Optional Optional<T>类(java.util.Optional)是一个容器类,代表一 ...

  7. 乐字节-Java8新特性-接口默认方法

    总概 JAVA8 已经发布很久,而且毫无疑问,java8是自java5(2004年发布)之后的最重要的版本.其中包括语言.编译器.库.工具和JVM等诸多方面的新特性. Java8 新特性列表如下: 接 ...

  8. 乐字节-Java8新特性之Base64和重复注解与类型注解

    上一篇小乐给大家说了<乐字节-Java8新特性之Date API>,接下来小乐继续给大家说一说Java8新特性之Base64和重复注解与类型注解. 一.Base64 在Java 8中,内置 ...

  9. 乐字节-Java8新特性之Date API

    上一篇文章,小乐给大家带来了Java8新特性之Optional,接下来本文将会给大家介绍Java8新特性之Date API 前言: Java 8通过发布新的Date-Time API来进一步加强对日期 ...

随机推荐

  1. 原型设计Axure的基本使用

    Axure是一款专业的原型设计工具, 让负责定义需求设计:功能和界面的人员能快速设计出所需产品,其中不仅包含了对软件产品的界面,交互逻辑的原型设计,还包含了流程图:web网站的线框图,并且能导出说明文 ...

  2. db2循环

    db2普通循环结构 while循环 while 条件 do 循环体 end while; LOOP循环 SET V_INDEX = 0; AUTHLOOP: LOOP V_INDEXV_INDEX = ...

  3. C&num; 循环语句 for循环

    循环:反复执行某段代码. 循环四要素:初始条件,循环条件,循环体,状态改变.for(初始条件;循环条件;状态改变){ 循环体} 给出初始条件,先判断是否满足循环条件,如果不满足条件则跳过for语句,如 ...

  4. python 调用mysql存储过程返回结果集

    存储过程: delimiter | ),)) begin select * from tb_test where mid = imid and user = iuser; end; | delimit ...

  5. AndroidManifest&period;xml文件综合详解(转)

    一,重要性AndroidManifest.xml是Android应用程序中最重要的文件之一.它是Android程序的全局配置文件,是每个 android程序中必须的文件.它位于我们开发的应用程序的根目 ...

  6. 【JavaScript】出现即使设置了ID也获取不到的可能原因与window&period;onload

    有时候.在JavaScript中.即使设置了ID也有可能出现document.getElementById()获取不到的情况,然后你就開始想document是否写错之类的.事实上根本就不是你的代码的大 ...

  7. bzoj 1912 &colon; &lbrack;Apio2010&rsqb;patrol 巡逻 树的直径

    题目链接 如果k==1, 显然就是直径. k==2的时候, 把直径的边权变为-1, 然后在求一次直径. 变为-1是因为如果在走一次这条边, 答案会增加1. 学到了新的求直径的方法... #includ ...

  8. 搭建web服务器-tomcat&plus;apache&plus;mysql&plus;eclipse

    1. 下载并安装jdk 注意环境变量的配置: java_home: jdk的路径  path:.;%java_home%\bin;%java_home%\jre\bin classpath :  .; ...

  9. Golang 入门系列&lpar;十&rpar; mysql数据库的使用

    之前,已经讲过一些Golang的基础的东西,感兴趣的可以看看以前的文章,https://www.cnblogs.com/zhangweizhong/category/1275863.html, 今天简 ...

  10. linux 学习笔记 cpio命令

    1 文件或目录打包 打包有如下多种情况 A>包含子目录打包 find /usr/lib -print /cpio -o >/uo/temp1.cpio 将/usr/lib目录下的文件与子目 ...