Java优雅解决空指针问题源码级别刨析Optional 2

简介: Java优雅解决空指针问题源码级别刨析Optional

2.2.5 flatMap()应用&源码解析

刚才已经通过map()获取了学生的姓名,操作非常简单。但是当产生链路获取时,map可以使用么?

如:学生->工作->公司->公司名称。现在可能大家脑袋里已经有了一个想法,就是通过map(),代

码结构如下:

studentOptional.map(Student::getJob).map(Job:
:getCompany).map(Company::getName);

但是这段代码是无法通过编译的。因为根据map的学习,每一次在调用的时候,都会对Optional的泛型进行改变,最终产生多层Optional嵌套的结构。如下图所示:

aa3ead5980974f6abb0f8b3ac1e7bef0.png

对于这个问题的解决,Optional类中提供了另外一个获取值的方法flatMap()。它本身用于多层调用,同时对于结果它不会形成多个Optional,而是将结果处理成最终的一个类型的Optional。但是通过flatMap获取的返回值必须是Optional类型。而map则没有这个限制。

2db388d1d0c24b04aeda30061a8fd901.png

使用示例如下:

Optional<String> nameOptional =
studentOptional.flatMap(Student::getJob).flat
Map(Job::getCompany).map(Company::getName);

2.2.6 filter()应用&源码解析

在获取某个对象中的属性值时,经常会根据特定的条件进行获取。之前的编码方法通常为:

Company company = optional().get();
if("oldlu".equals(company.getName)){
  sout(company);
}

Optional类中也提供了数据过滤的方法filter()来实现这个需求。其会根据传入的条件进行判断,如果匹配则返回一个Optional对象并包含对应的值,否则返回一个空值的Optional

使用示例如下:

Optional<Company> company =
companyOptional.filter(c ->
"oldlu".equals(c.getName()));

2.2.7 orElse()应用&源码解析

在取值的时候,如果值不存在,有时我们会考虑返回一个默认值。该需求就可以通过orElse()实现。

其内部会判断值是否为null,如果不为null,则返回该值,如果为null,则返回传入的默认值。

使用示例如下:

public class Demo1 {
  public static void
getCompanyName(Student student) {
    Optional<Student> studentOptional =
Optional.ofNullable(student);
    if (studentOptional.isPresent()) {
       String value =
studentOptional.flatMap(Student::getJob).fla
tMap(Job::getCompany).map(Company::getName).
orElse("default value");
      System.out.println(value);
   }
 }
  public static void main(String[] args) {
    Company company = new Company();
    //company.setName("oldlu");
    Optional<Company> companyOptional =
Optional.of(company);
    Job job = new Job();
    job.setName("pm");
    job.setCompany(companyOptional);
    Optional<Job> jobOptional =
Optional.of(job);
    Student s1 = new Student();
    s1.setName("张三");
    s1.setJob(jobOptional);
    getCompanyName(s1);
 }
 }

2.2.8 orElseGet()应用&源码解析

orElseGet()也是用于当Optional中没有值时,返回默认值的方法。但是它与orElse()的区别在于,它是延迟加载的。只有当Optional中没有值是才会被调用。

区别示例如下:

1)当公司名称不存在

public class Demo1 {
  public static void
getCompanyName(Student student) {
    Optional<Student> studentOptional =
Optional.ofNullable(student);
    if (studentOptional.isPresent()) {
      String value1 =
studentOptional.flatMap(Student::getJob).fla
tMap(Job::getCompany).map(Company::getName).
orElse(get("a"));
      String value2 =
studentOptional.flatMap(Student::getJob).fla
tMap(Job::getCompany).map(Company::getName).
orElseGet(()->get("b"));
      System.out.println("a: 
"+value1);
      System.out.println("b: 
"+value2);
   }
 }
  public static String get(String name) {
    System.out.println(name + "执行了方
法");
    return "exec";
 }
  public static void main(String[] args) {
    Company company = new Company();
    //company.setName("oldlu");
    Optional<Company> companyOptional =
Optional.of(company);
    Job job = new Job();
    job.setName("pm");
    job.setCompany(companyOptional);
    Optional<Job> jobOptional =
Optional.of(job);
    Student s1 = new Student();
s1.setName("张三");
    s1.setJob(jobOptional);
    getCompanyName(s1);
 }
}

执行结果

a执行了方法
b执行了方法
a:  exec
b:  exec


根据上述结果可知,当公司名称不存在时,orElse()与orElseGet()都被执行了。

2)公司名称存在

public class Demo1 {
  public static void
getCompanyName(Student student) {
    Optional<Student> studentOptional =
Optional.ofNullable(student);
    if (studentOptional.isPresent()) {
      String value1 =
studentOptional.flatMap(Student::getJob).fla
tMap(Job::getCompany).map(Company::getName).
orElse(get("a"));
      String value2 =
studentOptional.flatMap(Student::getJob).fla
tMap(Job::getCompany).map(Company::getName).
orElseGet(()->get("b"));
      System.out.println("a: 
"+value1);
      System.out.println("b: 
"+value2);
   }
 }
  public static String get(String name) {
    System.out.println(name + "执行了方
法");
    return "exec";
 }
  public static void main(String[] args) {
    Company company = new Company();
    company.setName("oldlu");
    Optional<Company> companyOptional =
Optional.of(company);
    Job job = new Job();
    job.setName("pm");
    job.setCompany(companyOptional);
    Optional<Job> jobOptional =
Optional.of(job);
    Student s1 = new Student();
    s1.setName("张三");
    s1.setJob(jobOptional);
    getCompanyName(s1);
 }
}

执行结果如下:

a执行了方法
a:  oldlu
b:  oldlu
目录
相关文章
|
3天前
|
自然语言处理 Java API
Java 8的Stream API和Optional类:概念与实战应用
【5月更文挑战第17天】Java 8引入了许多重要的新特性,其中Stream API和Optional类是最引人注目的两个。这些特性不仅简化了集合操作,还提供了更好的方式来处理可能为空的情况,从而提高了代码的健壮性和可读性。
26 7
|
2天前
|
监控 安全 NoSQL
采用java+springboot+vue.js+uniapp开发的一整套云MES系统源码 MES制造管理系统源码
MES系统是一套具备实时管理能力,建立一个全面的、集成的、稳定的制造物流质量控制体系;对生产线、工艺、人员、品质、效率等多方位的监控、分析、改进,满足精细化、透明化、自动化、实时化、数据化、一体化管理,实现企业柔性化制造管理。
20 3
|
3天前
|
存储 Java
Java基础复习(DayThree):字符串基础与StringBuffer、StringBuilder源码研究
Java基础复习(DayThree):字符串基础与StringBuffer、StringBuilder源码研究
Java基础复习(DayThree):字符串基础与StringBuffer、StringBuilder源码研究
|
3天前
|
数据采集 监控 安全
java数字工厂MES系统全套源码Java+idea+springboot专业为企业提供智能制造MES解决方案
"MES" 指的是制造执行系统(Manufacturing Execution System)。MES在制造业中扮演着至关重要的角色,它是位于企业资源计划(ERP)系统和车间控制系统之间的系统,用于实时收集、管理、分析和报告与制造过程相关的数据。
10 0
|
3天前
|
移动开发 监控 供应链
JAVA智慧工厂制造生产管理MES系统,全套源码,多端展示(app、小程序、H5、台后管理端)
一开始接触MES系统,很多人会和博主一样,对MES细节的应用不了解,这样很正常,因为MES系统相对于其他系统来讲应用比较多!
14 1
JAVA智慧工厂制造生产管理MES系统,全套源码,多端展示(app、小程序、H5、台后管理端)
|
3天前
|
NoSQL 算法 Java
【redis源码学习】持久化机制,java程序员面试算法宝典pdf
【redis源码学习】持久化机制,java程序员面试算法宝典pdf
|
4天前
|
存储 运维 Java
java云his系统源码一站式诊所SaaS系统Java版云HIS系统 八大特点
HIS系统采用面向技术架构的分析与设计方法,应用多层次应用体系架构设计,运用基于构件技术的系统搭建模式与基于组件模式的系统内核结构。通过建立统一接口标准,实现数据交换和集成共享,通过统一身份认证和授权控制,实现业务集成、界面集成。
29 1
|
4天前
|
Java
Java String 避免空指针的方法
Java String 避免空指针的方法
5 0
|
5天前
|
算法 Java 索引
【Java 刷题记录】双指针(下)
【Java 刷题记录】双指针
21 0
|
5天前
|
算法 Java 容器
【Java 刷题记录】双指针(上)
【Java 刷题记录】双指针
20 0
http://www.vxiaotou.com