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
目录
相关文章
|
1天前
|
人工智能 监控 Java
java互联网+智慧工地云平台SaaS源码
智慧工地以施工现场风险预知和联动预控为目标,将智能AI、传感技术、人像识别、监控、虚拟现实、物联网、5G、大数据、互联网等新一代科技信息技术植入到建筑、机械、人员穿戴设施、场地进出关口等各类设备中,实现工程管理与工程施工现场的整合
7 0
|
3天前
|
监控 Java BI
java基于云计算的SaaS医院his信息系统源码 HIS云平台源码
基于云计算技术的B/S架构的HIS系统源码,SaaS模式Java版云HIS系统,融合B/S版电子病历系统,支持电子病历四级,HIS与电子病历系统均拥有自主知识产权。
24 5
|
4天前
|
人工智能 监控 数据可视化
Java智慧工地云平台源码带APP SaaS模式 支持私有化部署和云部署
智慧工地是指应用智能技术和互联网手段对施工现场进行管理和监控的一种工地管理模式。它利用传感器、监控摄像头、人工智能、大数据等技术,实现对施工现场的实时监测、数据分析和智能决策,以提高工地的安全性、效率和质量(技术架构:微服务+Java+Spring Cloud +UniApp +MySql)。
19 4
|
5天前
|
Java 编译器 API
Java基础教程(17)-Java8中的lambda表达式和Stream、Optional
【4月更文挑战第17天】Lambda表达式是Java 8引入的函数式编程特性,允许函数作为参数或返回值。它有简洁的语法:`(parameters) -> expression 或 (parameters) ->{ statements; }`。FunctionalInterface注解用于标记单方法接口,可以用Lambda替换。
|
6天前
|
人工智能 监控 安全
JAVA基于SaaS模式的智慧工地云平台源码(云智慧工地解决方案)
智慧工地支持多端展示(PC端、手机端、平板端)SaaS微服务架构,项目监管端,工地管理端源码
13 0
|
6天前
|
搜索推荐 前端开发 Java
java医院绩效考核管理系统项目源码
系统需要和his系统进行对接,按照设定周期,从his系统获取医院科室和医生、护士、其他人员工作量,对没有录入信息化系统的工作量,绩效考核系统设有手工录入功能(可以批量导入),对获取的数据系统按照设定的公式进行汇算,且设置审核机制,可以退回修正,系统功能强大,完全模拟医院实际绩效核算过程,且每步核算都可以进行调整和参数设置,能适应医院多种绩效核算方式。
10 0
|
7天前
|
设计模式 算法 Java
[设计模式Java实现附plantuml源码~行为型]定义算法的框架——模板方法模式
[设计模式Java实现附plantuml源码~行为型]定义算法的框架——模板方法模式
|
7天前
|
设计模式 JavaScript Java
[设计模式Java实现附plantuml源码~行为型] 对象状态及其转换——状态模式
[设计模式Java实现附plantuml源码~行为型] 对象状态及其转换——状态模式
|
7天前
|
设计模式 存储 Java
[设计模式Java实现附plantuml源码~结构型]实现对象的复用——享元模式
[设计模式Java实现附plantuml源码~结构型]实现对象的复用——享元模式
|
7天前
|
设计模式 JavaScript Java
[设计模式Java实现附plantuml源码~结构型]处理多维度变化——桥接模式
[设计模式Java实现附plantuml源码~结构型]处理多维度变化——桥接模式