Java 8 新特性 Optional 类学习,理解并应用。NullPointerException空值检测

简介: JDK1.8开始引入的特性,Optional 类主要解决空指针异常(NullPointerException)问题。Optional类是一个可能包含或不包含非空值(可以为null)的容器对象。 如果一个值存在,调用 isPresent()方法将返回true、get()方法将返回该对象。 Optional类提供判断空值的方法,使用其中方法可以不用再显式地进行空值检测。

🍀Optional 类

JDK1.8开始引入的特性,Optional 类主要解决空指针异常(NullPointerException)问题。

Optional类是一个可能包含或不包含非空值(可以为null)的容器对象。 如果一个值存在,调用 isPresent()方法将返回true、get()方法将返回该对象。

Optional类提供判断空值的方法,使用其中方法可以不用再显式地进行空值检测。

举例:
User类的结构
在这里插入图片描述

String isocode = user.getAddress().getCountry().getIsocode().toUpperCase();

在避免出现空指针的情况下,我们常常需要将代码改造成这样:

if (user != null) {
    Address address = user.getAddress();
    if (address != null) {
        Country country = address.getCountry();
        if (country != null) {
            String isocode = country.getIsocode();
            if (isocode != null) {
                isocode = isocode.toUpperCase();
            }
        }
    }
}

在避免了空指针异常的情况下,代码变得冗长,且增加了之后的运维难度。

而利用Optional类,可以将这一判断过程变得赏心悦目。

重构下类,将getter方法修改成返回Optional引用

public class Address {
    private Country country;

    public Optional<Country> getCountry() {
        return Optional.ofNullable(country);
    }
    // ...
}

public class User {
    private Address address;
    
    public Optional<Address> getAddress() {
        return Optional.ofNullable(address);
    }
    // ...
@Test
public void whenChaining_thenOk() {
    User user = new User("berbai@88.com", "ber");

    String result = Optional.ofNullable(user)
      .flatMap(u -> u.getAddress())
      .flatMap(a -> a.getCountry())
      .map(c -> c.getIsocode())
      .orElse("default");

    assertEquals(result, "default");
}

通过方法引用可进一步简化代码

     String result2 = Optional.ofNullable(user)
             .flatMap(User::getAddress)
             .flatMap(Address::getCountry)
             .map(Country::getIsocode)
             .orElse("default");

🍀方法介绍

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

🍀Optional应用

创建 Optional 实例

// 创建一个空的Optional对象,此时调用emptyOpt 存在NoSuchElementException
Optional<User> emptyOpt = Optional.empty();
// 创建包含值的 Optional。注意:user必需不为空,否则NullPointerException
Optional<User> opt = Optional.of(user);
// 创建可能包含值或可能为空的 Optional。user可以为空
Optional<User> opt = Optional.ofNullable(user);

访问 Optional 对象的值

String name = "Ber";
Optional<String> opt = Optional.ofNullable(name);
// get() 如果 opt中有一个值,返回值,否则抛出 NoSuchElementException 。 
System.out.println(opt.get());  // 输出Ber,opt为空则NoSuchElementException
// 所以先判断opt是否空
opt.isPresent(); // 返回 true如果存在值,否则为 false 。
opt.ifPresent( u -> System.out.println(u)); // 只有 name 不为 null 的时候才会执行输出。

为空时返回默认值

  1. orElse()
String nameNull = null;
String name = "Ber";
// nameNull 为空,默认值则为name。
System.out.println(Optional.ofNullable(nameNull).orElse(name)); // 输出Ber

String name2 = "默认值";
// 如果对象初始值不为空,则默认值会被忽略
System.out.println(Optional.ofNullable(name).orElse(name2)); // 输出Ber,而不是默认值
  1. orElseGet()
// 在有值的时候返回值,如果为空,它会执行作为参数传入的 Supplier(供应者) 函数式接口,并将返回其执行结果
System.out.println(Optional.ofNullable(nameNull).orElseGet( () -> name)); // 输出Ber
  1. orElse() 和 orElseGet()的区别

在Optional对象为不为空时,两个方法都返回相应的非空值,但orElse()方法任创建了默认值对象,而orElseGet()方法不再创建。

为空返回异常

String nameNull = null;
// 这里设定为空返回IllegalArgumentException‎,返回异常可自行定义
Optional.ofNullable(nameNull).orElseThrow( () -> new ‎IllegalArgumentException‎());

Optional对象转换值

  1. map()

如果存在一个值,则应用提供的映射函数,如果结果不为空,则返回一个 Optional结果的 Optional 。

User user = new User("berbai@88.com", "ber");
String email = Optional.ofNullable(user).map(u -> u.getEmail()).orElse("default@gmail.com");
  1. flatMap()

如果一个值存在,应用提供的 Optional映射函数给它,返回该结果,否则返回一个空的 Optional 。

// 在User类增加返回 Optional 方法
public class User {    
    private String sex;

    public Optional<String> getSex() {
        return Optional.ofNullable(sex);
    }
    //...
}

User user = new User("berbai@88.com", "ber");
user.setSex("男")
String position = Optional.ofNullable(user).flatMap(u -> u.getSex()).orElse("女");

Optional过滤值

  1. filter()

如果一个值存在,并且该值给定的谓词相匹配时,返回一个 Optional描述的值,否则返回一个空的 Optional 。

User user = new User("berbai@88.com", "ber");
// email必须包含‘@’
Optional<User> result = Optional.ofNullable(user).filter(u -> u.getEmail() != null && u.getEmail().contains("@"));
目录
相关文章
|
1天前
|
Java
【Java基础】详解面向对象特性(诸如继承、重载、重写等等)
【Java基础】详解面向对象特性(诸如继承、重载、重写等等)
5 0
|
1天前
|
Java Nacos 开发者
Java从入门到精通:4.2.1学习新技术与框架——以Spring Boot和Spring Cloud Alibaba为例
Java从入门到精通:4.2.1学习新技术与框架——以Spring Boot和Spring Cloud Alibaba为例
|
1天前
|
Dubbo Java 应用服务中间件
Java从入门到精通:3.2.2分布式与并发编程——了解分布式系统的基本概念,学习使用Dubbo、Spring Cloud等分布式框架
Java从入门到精通:3.2.2分布式与并发编程——了解分布式系统的基本概念,学习使用Dubbo、Spring Cloud等分布式框架
|
1天前
|
SQL Java 数据库连接
Java从入门到精通:2.3.1数据库编程——学习JDBC技术,掌握Java与数据库的交互
ava从入门到精通:2.3.1数据库编程——学习JDBC技术,掌握Java与数据库的交互
|
1天前
|
设计模式 存储 前端开发
Java从入门到精通:2.2.1学习Java Web开发,了解Servlet和JSP技术,掌握MVC设计模式
Java从入门到精通:2.2.1学习Java Web开发,了解Servlet和JSP技术,掌握MVC设计模式
|
1天前
|
Java API
Java从入门到精通:2.1.5深入学习Java核心技术之文件操作
Java从入门到精通:2.1.5深入学习Java核心技术之文件操作
|
1天前
|
并行计算 算法 安全
Java从入门到精通:2.1.3深入学习Java核心技术——掌握Java多线程编程
Java从入门到精通:2.1.3深入学习Java核心技术——掌握Java多线程编程
|
1天前
|
缓存 Java
【Java基础】简说多线程(上)
【Java基础】简说多线程(上)
5 0
|
1天前
|
安全 Java 编译器
是时候来唠一唠synchronized关键字了,Java多线程的必问考点!
本文简要介绍了Java中的`synchronized`关键字,它是用于保证多线程环境下的同步,解决原子性、可见性和顺序性问题。从JDK1.6开始,synchronized进行了优化,性能得到提升,现在仍可在项目中使用。synchronized有三种用法:修饰实例方法、静态方法和代码块。文章还讨论了synchronized修饰代码块的锁对象、静态与非静态方法调用的互斥性,以及构造方法不能被同步修饰。此外,通过反汇编展示了`synchronized`在方法和代码块上的底层实现,涉及ObjectMonitor和monitorenter/monitorexit指令。
6 0
|
1天前
|
监控 安全 Java
在Java中如何优雅的停止一个线程?可别再用Thread.stop()了!
在Java中如何优雅的停止一个线程?可别再用Thread.stop()了!
7 2