Java8 Optional的详细使用教程

 更新时间:2021年2月21日 15:00  点击:1957

Optional介绍

Optional是Jdk1.8提供的一个新类,希望可以通过该类的引入解决令人烦躁的null判断问题,非常好用。个人理解:这个类是一个包装类,将要操作的java bean封装到该类的对象里面,同时将一些常用的判断逻辑封装为成员方法,结合lambda语法,实现比较优雅的链式调用。现在对Optional的API使用做一个简单的说明。

API介绍

Optional的所有的方法如下图所示,这些API大致可以分为以下几类:

1、构建API:构建一个Optional对象;方法有:empty( )、of( )、ofNullable( );

2、获取API:获取Optional对象里包装的值;方法有:get( )、orElse( )、orElseGet( )、orElseThrow( );

3、转换API:将Optional对象里包装的值转换成一个新的值;方法有:map( )、flatMap( );

4、判断API:对Optional对象里包装的值做一些逻辑判断;方法有:filter( )、isPresent( )、ifPresent( );

API使用方法

首先准备一个pojo类

@Data
@NoArgsConstructor
@AllArgsConstructor
@EqualsAndHashCode
@ToString
public class UserInfo {
 private String username;
 private String password;
 private Integer age;
 private String gender;
}

构建类

	@Test
 public void testBuildOptional() {
  // 构建一个value为null的optional对象;
  Optional<UserInfo> userInfoEmptyOpt = Optional.empty();

  // 构建一个value不可以为null的optional对象,如果of()的入参为null会报空指针异常;
  Optional<UserInfo> userInfoOpt = Optional.of(new UserInfo("阿飞","123456", 15, "man"));

  // 构建一个value可以为null的optional对象;
  Optional<UserInfo> userInfoNullOpt = Optional.ofNullable(null);

  System.out.println(userInfoEmptyOpt);
  System.out.println(userInfoOpt);
  System.out.println(userInfoNullOpt);
  
//  输出如下:
//  Optional.empty
//  Optional[UserInfo(username=阿飞, password=123456, age=15, gender=man)]
//  Optional.empty
 }

获取类

	@Test
 public void testGetOptionalValue() {
  Optional<UserInfo> userInfoEmptyOpt = Optional.empty();
  Optional<UserInfo> userInfoOpt = Optional.of(new UserInfo("阿飞","123456", 15, "man"));

  // 直接获取,注意如果value==null,会报NoSuchElementException异常
  UserInfo userInfo1 = userInfoOpt.get();
  // orElse可以传入一个UserInfo类型的对象作为默认值;当value!=null时,返回value值;当value==null时,返回默认值作为代替;
  UserInfo userInfo2 = userInfoEmptyOpt.orElse(new UserInfo("阿飞1","123456", 15, "man"));
  // orElseGet和orElse不同的是orElseGet可以传入一段lambda表达式;当value!=null时,返回value值;当value==null时,使用该lambda返回的对象作为默认值;
  UserInfo userInfo3 = userInfoEmptyOpt.orElseGet(() -> new UserInfo("阿飞2","123456", 15, "man"));
  // orElseThrow可以传入一段lambda表达式,lambda返回一个Exception;当value!=null时,返回value值;当value==null时,抛出该异常;
  UserInfo userInfo4 = userInfoOpt.orElseThrow(NullPointerException::new);

  System.out.println(userInfo1);
  System.out.println(userInfo2);
  System.out.println(userInfo3);
  System.out.println(userInfo4);

  // 输出如下:
  // UserInfo(username=阿飞, password=123456, age=15, gender=man)
  // UserInfo(username=阿飞1, password=123456, age=15, gender=man)
  // UserInfo(username=阿飞2, password=123456, age=15, gender=man)
  // UserInfo(username=阿飞, password=123456, age=15, gender=man)
 }

转换类

	@Test
 public void testMapOptionalValue() {
  Optional<UserInfo> userInfoOpt = Optional.of(new UserInfo("阿飞","123456", 15, "man"));

  // 原来value的类型是UserInfo,经过map转换为Optional<String>
  Optional<String> username = userInfoOpt.map(UserInfo::getUsername);

  // 当map的入参也是一个Optional时,经过map转化后会形成Optional<Optional<String>>这种嵌套结构;但flatMap可以把这种嵌套结构打平;
  Optional<Optional<String>> unFlatMap = userInfoOpt.map(user -> Optional.of(user.getUsername()));
  Optional<String> flatMap = userInfoOpt.flatMap(user -> Optional.of(user.getUsername()));

  System.out.println(username);
  System.out.println(unFlatMap);
  System.out.println(flatMap);
  
  // 输出如下:
  // Optional[阿飞]
  // Optional[Optional[阿飞]]
  // Optional[阿飞]
 }

判断类

	@Test
 public void testJudgeOptionalValue() {
  Optional<UserInfo> userInfoEmptyOpt = Optional.empty();
  Optional<UserInfo> userInfoOpt = Optional.of(new UserInfo("阿飞", "123456", 15, "man"));

  // filter传入一个lambda,lambda返回值为boolean;true:不做任何改变,false:返回一个空的optional;
  Optional<UserInfo> userInfo = userInfoOpt.filter(user -> "错误的密码".equals(user.getPassword()));
  System.out.println(userInfo);

  // isPresent就是判断value是不是null;我们在调用get之前,一定要先调用isPresent,因为直接如果value是null,直接调用get会报异常;
  if (userInfoEmptyOpt.isPresent()) {
   UserInfo value = userInfoEmptyOpt.get();
   System.out.println("optional value:" + value);
  } else {
   System.out.println("optional value==null");
  }

  // ifPresent传入一段lambda,当value!=null时,执行里面的逻辑;当当value==null时,啥都不干;
  userInfoOpt.ifPresent(value -> System.out.println("optional value:" + value));
  
  // 输入如下:
  // Optional.empty
  // optional value==null
  // optional value:UserInfo(username=阿飞, password=123456, age=15, gender=man)
 }

试一下这些API吧

package com.fly.optional;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import lombok.ToString;
import org.junit.Test;
import org.omg.CosNaming.NamingContextPackage.NotFound;

import java.math.BigDecimal;
import java.util.Optional;

public class OptionDemo {

 @Data
 @NoArgsConstructor
 @AllArgsConstructor
 @ToString
 static class Employee {
  private Long id;
  private String name;
  private Boolean leader;
 }

 @Data
 @NoArgsConstructor
 @AllArgsConstructor
 @ToString
 static class Leader {
  private Long employeeId;
  private BigDecimal bonus;
 }

 /**
  * 找到ID为1的员工,如果有奖金就打印出来,没有就打印没有奖金;
  * @throws NotFound
  */
 @Test
 public void tst() throws NotFound {
  Optional<Leader> leader = Optional.ofNullable(getEmployeeById(1L).filter(Employee::getLeader).map(Employee::getId).flatMap(this::getLeaderByEmployeeId).orElse(null));
  if (leader.isPresent()) {
   Optional.of(leader.map(Leader::getBonus).map(bonus -> String.format("员工ID为1的leader奖金为:%s", bonus)).orElse("员工ID为1的leader也没有奖金")).ifPresent(System.out::println);
  } else {
   System.out.println("员工ID为1的leader未找到,他可能只是一个基层员工,不配拥有奖金");
  }
 }

 private Optional<Employee> getEmployeeById(Long id) {
  //return Optional.of(new Employee(1L, "大老板", Boolean.TRUE));
  return Optional.of(new Employee(1L, "大老板", Boolean.FALSE));
 }

 private Optional<Leader> getLeaderByEmployeeId(Long employeeId) {
  //return employeeId == 1L ? Optional.of(new Leader(1L, BigDecimal.valueOf(1000000000))) : Optional.empty();
  return employeeId == 1L ? Optional.of(new Leader(1L, null)) : Optional.empty();
 }
}

总结

到此这篇关于Java8 Optional的详细使用的文章就介绍到这了,更多相关Java8 Optional使用内容请搜索猪先飞以前的文章或继续浏览下面的相关文章希望大家以后多多支持猪先飞!

[!--infotagslink--]

相关文章

  • Java8 实现stream将对象集合list中抽取属性集合转化为map或list

    这篇文章主要介绍了Java8 实现stream将对象集合list中抽取属性集合转化为map或list的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-02-05
  • java8如何用Stream查List对象某属性是否有重复

    这篇文章主要介绍了java8如何用Stream查List对象某属性是否有重复的操作,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...2021-09-11
  • java8中的Collectors.groupingBy用法详解

    这篇文章主要介绍了java8中的Collectors.groupingBy用法详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2021-03-17
  • 图解PHP使用Zend Guard 6.0加密方法教程

    有时为了网站安全和版权问题,会对自己写的php源码进行加密,在php加密技术上最常用的是zend公司的zend guard 加密软件,现在我们来图文讲解一下。 下面就简单说说如何...2016-11-25
  • ps怎么使用HSL面板

    ps软件是现在很多人都会使用到的,HSL面板在ps软件中又有着非常独特的作用。这次文章就给大家介绍下ps怎么使用HSL面板,还不知道使用方法的下面一起来看看。 &#8195;...2017-07-06
  • Plesk控制面板新手使用手册总结

    许多的朋友对于Plesk控制面板应用不是非常的了解特别是英文版的Plesk控制面板,在这里小编整理了一些关于Plesk控制面板常用的使用方案整理,具体如下。 本文基于Linu...2016-10-10
  • Java8处理List的双层循环问题

    这篇文章主要介绍了Java8处理List的双层循环问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-08-19
  • 使用insertAfter()方法在现有元素后添加一个新元素

    复制代码 代码如下: //在现有元素后添加一个新元素 function insertAfter(newElement, targetElement){ var parent = targetElement.parentNode; if (parent.lastChild == targetElement){ parent.appendChild(newEl...2014-05-31
  • 使用GruntJS构建Web程序之构建篇

    大概有如下步骤 新建项目Bejs 新建文件package.json 新建文件Gruntfile.js 命令行执行grunt任务 一、新建项目Bejs源码放在src下,该目录有两个js文件,selector.js和ajax.js。编译后代码放在dest,这个grunt会...2014-06-07
  • 使用percona-toolkit操作MySQL的实用命令小结

    1.pt-archiver 功能介绍: 将mysql数据库中表的记录归档到另外一个表或者文件 用法介绍: pt-archiver [OPTION...] --source DSN --where WHERE 这个工具只是归档旧的数据,不会对线上数据的OLTP查询造成太大影响,你可以将...2015-11-24
  • 如何使用php脚本给html中引用的js和css路径打上版本号

    在搜索引擎中搜索关键字.htaccess 缓存,你可以搜索到很多关于设置网站文件缓存的教程,通过设置可以将css、js等不太经常更新的文件缓存在浏览器端,这样访客每次访问你的网站的时候,浏览器就可以从浏览器的缓存中获取css、...2015-11-24
  • 安装和使用percona-toolkit来辅助操作MySQL的基本教程

    一、percona-toolkit简介 percona-toolkit是一组高级命令行工具的集合,用来执行各种通过手工执行非常复杂和麻烦的mysql和系统任务,这些任务包括: 检查master和slave数据的一致性 有效地对记录进行归档 查找重复的索...2015-11-24
  • jQuery 1.9使用$.support替代$.browser的使用方法

    jQuery 从 1.9 版开始,移除了 $.browser 和 $.browser.version , 取而代之的是 $.support 。 在更新的 2.0 版本中,将不再支持 IE 6/7/8。 以后,如果用户需要支持 IE 6/7/8,只能使用 jQuery 1.9。 如果要全面支持 IE,并混合...2014-05-31
  • 浅谈Java8 的foreach跳出循环break/return

    这篇文章主要介绍了Java8 的foreach跳出循环break/return,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教...2021-07-28
  • MySQL日志分析软件mysqlsla的安装和使用教程

    一、下载 mysqlsla [root@localhost tmp]# wget http://hackmysql.com/scripts/mysqlsla-2.03.tar.gz--19:45:45-- http://hackmysql.com/scripts/mysqlsla-2.03.tar.gzResolving hackmysql.com... 64.13.232.157Conn...2015-11-24
  • php语言中使用json的技巧及json的实现代码详解

    目前,JSON已经成为最流行的数据交换格式之一,各大网站的API几乎都支持它。我写过一篇《数据类型和JSON格式》,探讨它的设计思想。今天,我想总结一下PHP语言对它的支持,这是开发互联网应用程序(特别是编写API)必须了解的知识...2015-10-30
  • PHP实现无限级分类(不使用递归)

    无限级分类在开发中经常使用,例如:部门结构、文章分类。无限级分类的难点在于“输出”和“查询”,例如 将文章分类输出为<ul>列表形式; 查找分类A下面所有分类包含的文章。1.实现原理 几种常见的实现方法,各有利弊。其中...2015-10-23
  • C#注释的一些使用方法浅谈

    C#注释的一些使用方法浅谈,需要的朋友可以参考一下...2020-06-25
  • php类的使用实例教程

    php类的使用实例教程 <?php /** * Class program for yinghua05-2 * designer :songsong */ class Template { var $tpl_vars; var $tpl_path; var $_deb...2016-11-25
  • 双冒号 ::在PHP中的使用情况

    前几天在百度知道里面看到有人问PHP中双冒号::的用法,当时给他的回答比较简洁因为手机打字不大方便!今天突然想起来,所以在这里总结一下我遇到的双冒号::在PHP中使用的情况!双冒号操作符即作用域限定操作符Scope Resoluti...2015-11-08