Optional--Java8新特性最佳实践

news2024/11/24 12:56:51

Optional是在 Java8中引入的新特性之一。使用Optional类包装数据,可以避免经典的空检查和一些try-catch代码块。也能够通过链式方法调用,写出更流畅的函数式编程的代码。另一方面,滥用Optional也会导致性能低下和代码混乱。过往项目业务中有大量使用到,现做个归纳总结如下。

//消息实体转换
refundMQModel = Optional.ofNullable(message)
    .map(e -> JsonUtil.fromson(e, RefundMQModel.class )) 
    .filter(e -> e.smsOrderNo != null)
    .orElseThrow(()->new VerifyException(message ,"The smsOrderNo does not exist"));

//获取订单信息
SmsFlowModel smsFlowModel = RefundUtil.queryOrder(refundMOModel.smsOrderNo)
    .orElseThrow(()-> new VerifyException(message ,"Order info does not exist"));

//是否需要退费
RefundUtil.whetherToRefund(smsFlowModel).filter(e ->e.equals(Boolean.TRUE))
    .orElseThrow(()->new VerifyException(message ," No refund required"));

//获取退费记录,不存在则自动创建 
RefundDBModel dbRefundMQModel =
        RefundUtil.getOrCreateRefundRecord(smsFlowModel.getSmsOrderNo())
        .map(e -> e.withOrderInfo(smsFlowModel))
        .orElseThrow(()-> new BusinessException(message ,"Failed to get the refund record"));

//调用外部退费接口
Optional.ofNullable(dbRefundMQModel)
    .filter(e -> e.isNeedsCallOperator())
    .ifPresent(refund ->{
             //状态机(乐观锁)&&变更状态为处理中
             RefundUtil.update0peratorStatusByOrderNo(refund .getSms0rderNo(),
                     OperatorRefundStatusEnum.REFUNDING.getValue(), 
                     OperatorRefundStatusEnum.TO_BE_REFUND.getValue());
                    //调用退费接口&&变更为已完成
                     RefundUtil.callOperatorRefund(refund);
            });

//开启平台内退费
Optional.ofNullable(dbRefundMQModel)
        .filter(e -> e.isNeedsToRefund())
        .ifPresent(refund ->{
                 //状态机(乐观锁)&&变更状态为处理中
                 RefundUtil.updateRefundStatusByOrderNo(refund.getSmsOrderNo(),
                         RefundStatusEnum.REFUNDING.getValue(),
                         OperatorRefundStatusEnum.TO_BE_REFUND.getValue());
                //平台正式退费&&变更为已完成
                 RefundUtil . callPlatformRefund ( refund );
        });

1. empty()

返回一个Optional容器对象,而不是 null。

public Optional<User> getUser(String name) {
    if (StringUtil.isNotEmpty(name)) {
        return RemoteService.getUser(name);
    } 
    return Optional.empty();
}

2. orElse()

返回Optional中包装的值,取不到值时返回指定的 default。

public static final String USER_STATUS = "UNKNOWN";
...
public String findUserStatus(long id) {
    Optional<String> status = ... ; // 
    return status.orElse(USER_STATUS);
}

//不要这么写
public String findUserStatus(long id) {
    Optional<String> status = ... ; // 
    return status.orElse("UNKNOWN");//这样每次都会新建一个String对象
}

orElse()无论如何都会执行括号中的内容。适用于返回静态资源、字符串等。远程调用,或者涉及大量的文件 IO,代价巨大。

3. orElseGet()

返回Optional中包装的值,取不到值时返回指定的 default。

public String getName() {
    System.out.print("method called");
}
String name1 = Optional.of("String").orElse(getName()); //output: method called
String name2 = Optional.of("String").orElseGet(() -> getName()); //output:

3. orElseThrow()

针对阻塞性的业务场景使用比较合适,例如没有从上游获取到用户信息,下面的所有操作都无法进行,那此时就应该抛出异常。正常的写法是先判空,再手动 throw 异常。

public String findUser(long id){
    Optional<User> user = remoteService.getUserById(id);
    return user.orElseThrow(IllegalStateException::new);
}

4. ifPresent()

不为空则执行时,使用 ifPresent()。没有性能上的优势,只是使代码更简洁。

Optional<Integer> status = Optional.ofNullable(tokenService.getUserId("id"));
if (status.isPresent()) {
    System.out.println("Status: " + status.get());
}
//现在
status.ifPresent(System.out::println);

5.使用示例如下

//传统写法
public Account getAccountClassic() {
    Account account = accountRepository.get("jack");
    if(account == null) {
        throw new AccountNotFound();
    }
    return account;
}

// Optional写法
public Account getAccountOptional() {
    return accountRepository.find("jack")
                            .orElseThrow(AccountNotFound::new);
}
//传统写法
public AccountHolder getAccountHolderClassic() {
    Account account = accountRepository.find("jack");
    if (account == null) {
        throw new AccountNotFound();
    }
    return account.getHolder();
}

// Optional写法
public AccountHolder getAccountHolder() {
    return accountRepository.find("jack")
                            .map(Account::getHolder)
                            .orElseThrow(AccountNotFound::new);
}

6. 注意不要滥用

  • 使用Optional包装简单的局部变量

在Optional中包装变量只是为了利用它的API进行简单操作。反例

Optional.ofNullable(account)
        .ifPresent(acc-> processAccount(acc));

这里使用Optional没有带来任何价值。在这种情况下,我们应该使用经典的空检查:

if (account != null) {
    processAccount(account);
}

使用经典的方式,可以增加代码的可读性,也会减少Optional对象的创建。

  • 使用Optional包装字段

使用Optional包装字段,将导致在不需要对象的地方创建对象,如果重复使用,则会导致性能下降。此外,Optional包装的字段也不能进行序列化,因此,使用Optional字段可能会导致序列化问题。

一般来说,对于 POJO 中的 getter,更适合返回实际类型,而不是Optional类型。特别是,对于实体 bean、data module和 DTO 来说,建议都使用传统的 getter。反例:

1.作为序列化类中的字段
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class Account implements Serializable {
    @IdprivateString id;
    private Optional<String> number;
    privateString holder;
}

// 执行序列化操作
public static void main(String[] args) throws IOException {
    new ObjectOutputStream(new ByteArrayOutputStream())
        .writeObject(Account
                        .builder()
                        .number(Optional.of("123"))
                        .build());
}

// 将会抛出异常
Exception in thread "main" java.io.NotSerializableException: java.util.Optional
	at java.base/java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1187)
	at java.base/java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1572)
	at java.base/java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1529)
	at java.base/java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1438)
	at java.base/java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1181)
	at java.base/java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:350)
	at com.example.demo.DemoApplication.main(DemoApplication.java:18)
2..Json转换
publicclass Account implements Serializable {
    private String number;
    public Optional<String> getNumber() {
        return Optional.ofNullable(number);
    }
    publicvoid setNumber(Stringnumber) {
        this.number = number;
    }
}

使用Json序列化时,我们得到的结果是:

{"number":{"present":true}}

但是我们期望的结果是:

{"number":"123456"}

JPA entity中使用Optional字段

@Data
@Entity
public class Account implements Serializable{
    @Idprivate Long id;
    private Optional<String> number;
}
3.启动时,spring boot会直接报错
2023-03-05T08:08:26.103+08:00 ERROR 68105 --- [           main] o.s.boot.SpringApplication               : Application run failed
org.springframework.beans.factory.BeanCreationException: Error creating bean withname'entityManagerFactory' defined inclasspathresource [org/springframework/boot/autoconfigure/orm/jpa/HibernateJpaConfiguration.class]: Could not determine recommended JdbcType for`java.util.Optional<java.lang.String>`
	at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.initializeBean(AbstractAutowireCapableBeanFactory.java:1752) ~[spring-beans-6.0.5.jar:6.0.5]
	at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:600) ~[spring-beans-6.0.5.jar:6.0.5]
    xxx
    xxx
    xxx
  • Optional作为方法参数
public Account createAccount(Stringnumber, Optional<String> holder) {
    // todo save
}

上面例子中,参数holder可以为空。很显然,采用方法重载的方式要比这种方式更清晰。

public Account createAccount(Stringnumber)
public Account createAccount(Stringnumber, String holder)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/403752.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Linux】旋转锁 | 读写锁

在之前的线程学习中&#xff0c;用到的锁都是挂起等待锁&#xff0c;如果申请不到锁&#xff0c;那就会在锁中等待&#xff1b; 自旋锁则不大相似 文章目录1.自旋锁1.1 概念1.2 接口1.2.1 pthread_spin_init/destroy1.2.2 pthread_spin_lock1.2.3 pthread_spin_unlock2.读写锁…

VIO优化中不客观自由度 (gauge freedom) 的处理 (gauge handle)

文章目录1. 不可观的解释2. 几种不同的gauge handle处理方式2.1. free gauge方式2.2. fix gauge方式2.3. prior gauge方式2.4. g2o tutorial方式3.不同方式的协方差矩阵1. 不可观的解释 这篇论文 中对VIO的4-DOF不可观的定义如下&#xff0c;可以看到这种不可观就是如果对最后…

gerrit操作和jinkens编译合入代码

gerrit 先 查看自己的push 找到后添加reviewer 填写邮箱开头就可以出来 记得1 然后send 让人review 编译不过&#xff0c;gerrit上查看 1.是不是checkstyle问题 2.编译不过&#xff0c;去jinkens查看 先retrigger重新编译 如果发现多次编译失败 则要看下console output 查…

【ONE·Data || 顺序表】

总言 数据结构基础&#xff1a;顺序表模拟实现。    文章目录总言1、顺序表各接口功能实现描述1.1、如何创建一个顺序表&#xff1f;1.2、如何初始化顺序表&#xff1a;SLInit1.3、顺序表的尾插、头插1.3.1、顺序表尾插1.0&#xff1a;SLPushBack1.3.2、顺序表头插1.0&#x…

网络连接的三种模式

文章目录前言一、三种连接模式介绍二、三种网络连接模式的区别前言 在进行虚拟机配置时&#xff0c;网络连接分为三种模式&#xff1a;桥接模式&#xff0c;NAT模式&#xff0c;主机模式 一、三种连接模式介绍 张三、李四、王五在同一个网段&#xff0c;所以他们之间可以相互…

数据结构---双链表

专栏&#xff1a;数据结构 个人主页&#xff1a;HaiFan. 专栏简介&#xff1a;从零开始&#xff0c;数据结构&#xff01;&#xff01; 双链表前言双链表各接口的实现为要插入的值开辟一块空间BuyLN初始化LNInit和销毁LNDestory打印链表中的值LNPrint尾插LNPushBack和尾删LNPop…

vue2+elementUI完成添加学生删除学生案列

效果图&#xff1a; 点击添加学生按钮&#xff0c;弹出Dialog,收集用户信息&#xff1a; el-table中自定义复选框&#xff0c;选中一行&#xff0c;可以点击删除 代码区域&#xff1a;就一个HTML文件 <!DOCTYPE html> <html lang"en"> <head>&…

Flume基操

Flume概述 Flume 定义 Flume 是 Cloudera 提供的一个高可用的&#xff0c;高可靠的&#xff0c;分布式的海量日志采集、聚合和传输的系统。Flume 基于流式架构&#xff0c;灵活简单。 Flume最主要的作用就是&#xff0c;实时读取服务器本地磁盘的数据&#xff0c;将数据写入到…

带恒温冷藏功能的便携式自动采样器——可用于毒情监测

污水采样在验毒的工作流程中是怎样进行的呢&#xff1f; 污水采样&#xff1a;每个季度采样一次。例如在某市48家污水处理厂54个进水口采取水样&#xff0c;用便携式水质自动采样器连续采样7天&#xff0c;一天采样12次成为一个混合样。也就是说&#xff0c;一次采样的话&…

如何在 VS Code 中安装运行、编写C语言程序

1.下载 安装VS Code 去官网下载&#xff1a;https://code.visualstudio.com/Download 直接下载&#xff0c;安装即可。 2.安装VS code中2个插件 打开软件运行&#xff0c;在扩展商店中分别搜索安装 C/C 和 code runner 插件 3.下载mingb64 官网下载 https://sourceforge…

消息队列MQ用来做什么的,市场上主流的四大MQ如何选择?RabbitMQ带你HelloWorld!

文章目录MQ用来做什么的MQ会有什么样的麻烦MQ消息队列模式分类MQ消息队列常用协议市场主流四大MQRabbitMQ项目开发RabbitMQ中的组成部分MQ用来做什么的 省流 &#xff1a;系统解耦、异步调用、流量削峰 系统解耦 首先举例下面这个场景&#xff0c;现有ABCDE五个系统&#xff…

小黑子—Java从入门到入土过程:第二章

Java零基础入门2.0Java系列第二章1. 注释和关键字2. 字面量3. 变量3.1 基本用法3.2 使用方式3.3 注意事项4. 变量练习5. 计算机中的数据存储5.1 计算机的存储规则5.2 进制5.3 进制间转换二进制转十八进制转十十六进制转十十进制转其他进制6. 数据类型7. 定义变量的练习8. 标识符…

MATLAB——将直接型转化为并联型和级联型

题目1(IIR)&#xff1a; 已知一个系统的传递函数为&#xff1a; H&#xff08;z&#xff09;8−4z−111z−2−2z−31−1.25z−10.75z−2−0.125z−3H&#xff08;z&#xff09;\frac{8-4z^{-1}11z^{-2}-2z^{-3}}{1-1.25z^{-1}0.75z^{-2}-0.125z^{-3}}H&#xff08;z&#xff09…

Leedcode 1137. 第 N 个泰波那契数

泰波那契序列 Tn 定义如下&#xff1a; T0 0, T1 1, T2 1, 且在 n > 0 的条件下 Tn3 Tn Tn1 Tn2 给你整数 n&#xff0c;请返回第 n 个泰波那契数 Tn 的值。 示例 1&#xff1a; 输入&#xff1a;n 4 输出&#xff1a;4 解释&#xff1a; T_3 0 1 1 2 T_4 1 …

2.线性表的顺序表示

数据结构很重要&#xff01; 数据结构很重要&#xff01;&#xff01;&#xff01; 数据结构很重要&#xff01;&#xff01;&#xff01;&#xff01; 思考 1.线性表的顺序表示内容有哪些&#xff1f;&#xff08;What&#xff09; 2.为什么要学线性表的顺序表示? ? (Why)…

POI 操作Excel的单元格样式超过64000的异常问题解决

文章目录POI 操作Excel的单元格样式超过64000的异常问题解决问题描述问题原因问题分析和解决简单的Excel文件生成Demo最终的解决方案POI 操作Excel的单元格样式超过64000的异常问题解决 问题描述 在用POI 生成Excel文件时&#xff0c;如果自定义的单元格的样式超过64000行&am…

SpringBoot+WebSocket实时监控异常

# 写在前面此异常非彼异常&#xff0c;标题所说的异常是业务上的异常。最近做了一个需求&#xff0c;消防的设备巡检&#xff0c;如果巡检发现异常&#xff0c;通过手机端提交&#xff0c;后台的实时监控页面实时获取到该设备的信息及位置&#xff0c;然后安排员工去处理。因为…

2.9.1 Packet Tracer - Basic Switch and End Device Configuration(作业)

Packet Tracer - 交换机和终端设备的基本 配置地址分配表目标使用命令行界面 (CLI)&#xff0c;在两台思科互联网络 操作系统 (IOS) 交换机上配置主机名和 IP 地址。使用思科 IOS 命令指定或限制对设备 配置的访问。使用 IOS 命令来保存当前的运行配置。配置两台主机设备的 IP …

JavaScript Boolean(布尔)对象

Boolean&#xff08;布尔&#xff09;对象用于将非布尔值转换为布尔值&#xff08;true 或者 false&#xff09;&#xff0c;是三种包装对象&#xff1a;Number、String和Boolean中最简单的一种&#xff0c;它没有大量的实例属性和方法。在线实例检查布尔值检查布尔对象是 true…

大数据-玩转数据-mysql规范

整体图谱 正文部分 一、数据库命令规范 所有数据库对象名称必须使用小写字母并用下划线分割 所有数据库对象名称禁止使用mysql保留关键字&#xff08;如果表名中包含关键字查询时&#xff0c;需要将其用单引号括起来&#xff09; 数据库对象的命名要能做到见名识意&#xff…