Java中的Optional类
所望 2024-10-03 12:35:01 阅读 83
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档
文章目录
前言一、Optional是什么?二、Optional对象的方法1.isPresent、isEmpty方法2.empty、of、ofNullable方法3.get、orElse、orElseGet与orElseThrow方法4.ifPresent、ifPresentOrElse、filter方法5.Map与flatMap方法6.Stream方法
三、不适合使用Optional对象的情况总结
前言
java中尝试访问空引用的属性调用空引用的方法是会报空指针NullPointerException异常。实际项目中会处理大量为空的值,代码会有很多的条件判断,难以阅读与维护。
<code> if(user!=null){
System.out.println(user.getFullName());
}else {
User defaultUser = new User("Stark", "Tony Stark");
System.out.println(defaultUser.getFullName());
}
本笔记为总结视频内容中的Java中的Optional类的使用方法。
视频链接:https://www.bilibili.com/video/BV1dc411X7nW
一、Optional是什么?
Optional类引入了一种显式的方式来处理可能为空的对象,强制程序员在可能为空的情况下进行显式的处理,以避免空指针异常。
Optional类似容器,可以包含各种类型的值,也可以为null。Optional类提供了一系列方法来方便地操作内部的值。常用的方法有get、orElse、orElseGet、orElseThrow等。
Optional的设计也考虑了函数式编程的原则,可以与Lambda表达式和StreamAPI等特性结合使用,可以进行链式调用替代命令式编程的方式通过编写if条件语句检查null值。
二、Optional对象的方法
首先我们需要创建两个类,User类与UserRepository类。
User类
import java.util.Optional;
public class User {
String name;
String fullName;
public User(String name, String fullName) {
this.name = name;
this.fullName = fullName;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getFullName() {
return fullName;
}
public void setFullName(String fullName) {
this.fullName = fullName;
}
}
UserRepository类
import java.util.Optional;
public class UserRepository {
public User findUserByName(String name){
if(name.equals("Peter")){
return new User("Peter","Peter Parker");
}
else {
return null;
}
}
}
1.isPresent、isEmpty方法
isPresent方法用于检查optional内是否存在值,返回为布尔值,存在为true,不存在为false。
isEmpty方法用于检查optional内是否为空,返回为布尔值,为空为true,不为空为false。
//创建值为null的Optional对象
Optional<Object> optionalBox = Optional.empty();
System.out.println(optionalBox.isPresent());
System.out.println(optionalBox.isEmpty());
返回值为false与true。
2.empty、of、ofNullable方法
若创建的对象为null,则可以采用empty()方法。
Optional<Object> optionalBox = Optional.empty();
创建不为null的对象需要调用of方法,此时若value为null,则抛出NullPointerException异常。
String value = "Peter";
Optional<String> optionalBox = Optional.of(value);
若想要创建可能为null的对象,可以使用ofNullable( )方法。
String value = "Peter";
Optional<String> optionalBox = Optional.ofNullable(value);
3.get、orElse、orElseGet与orElseThrow方法
取值可以使用get()方法,若取值对象为null,会报java.util.NoSuchElementException异常(并非java.lang.NullPointerException异常)。
String value2 = optionalBox.get();
System.out.println(value2);
我们这里修改UserRepository类中的findUserByName方法,使其返回值为Optional对象。
public Optional<User> findUserByName(String name){
if(name.equals("Peter")){
return Optional.of(new User("Peter","Peter Parker"));
}
else {
return Optional.empty();
}
}
orElse是Optional类中的一个重要方法,它用于获取值或在值为空的情况下提供一个默认值。
UserRepository userRepository = new UserRepository();
Optional<User> optionalUser = userRepository.findUserByName("Peter2");
User user = optionalUser.orElse(new User("Stark","Tony Stark"));
System.out.println(user.getFullName());
orElse方法即使读到的数据不为null,仍会创建一个新User对象。因此最好使用只有读到的数据为null的时候才会新建对象的orElseGet方法。
orElseGet方法的参数为Supplier的函数式接口,需要使用Lambda表达式实现。
optionalUser.orElseGet(()->new User("Stark","Tony Stark"));
System.out.println(user.getFullName());
orElseThrow方法用于在 Optional 对象中的值为空时抛出一个指定的异常。
orElseThrow方法通过Supplier的函数式接口,可以生成自定义异常,默认抛出异常为java.util.NoSuchElementException。
UserRepository userRepository = new UserRepository();
Optional<User> optionalUser = userRepository.findUserByName("Peter2");
optionalUser.orElseThrow(()->new RuntimeException("User not found"));
4.ifPresent、ifPresentOrElse、filter方法
ifPresent方法参数中若对象不为null,则会执行Labmda中的方法;
若参数对象为null,则不会执行Labmda中的方法,也不会报错。
UserRepository userRepository = new UserRepository();
Optional<User> optionalUser = userRepository.findUserByName("Peter2");
optionalUser.ifPresent(user -> System.out.println(user.getFullName()));
当我们希望值为空时进行其他操作,需要使用ifPresentOrElse方法。
optionalUser.ifPresentOrElse(user -> System.out.println(user.getFullName()),
()->System.out.println("User not found"));
若满足filter方法中的条件,则会返回包含值的Optional对象,如果不满足,则返回空的Optional对象
Optional<User> optionalUser2 = optionalUser.filter(user -> user.getFullName().equals("Peter Parker"));
System.out.println(optionalUser2.isPresent());
5.Map与flatMap方法
进行两个方法前需要将User类中的 getFullName方法返回值修改为Optional。
public Optional<String> getFullName(){
return Optional.ofNullable(fullName);
}
map方法:对Optional中的值进行转换(若值为空,则map方法什么也不会做,直接返回空的Optional对象)。
这个变换基于提供该map的函数,并且这个变换是可选的,如果optional的值为空则不会做任何改变,并且map方法不会改变原始的Optional对象,而返回新的Optional对象,因此可以链式调用进行多个转换操作。
UserRepository userRepository = new UserRepository();
Optional<User> optionalUser = userRepository.findUserByName("Peter");
Optional<String> optionalFullName = optionalUser.map(User::getFullName);
System.out.println(optionalFullName.get());
flatmap方法用于扁平化嵌套的Optional结构,以避免引入不必要的嵌套层级,具体为flatmap的转换函数返回的必须是另一个Optional对象,意味着flatMap方法可以用于嵌套的Optional情况,可以将两个为嵌套关系的Optional对象转换为一个。如果原始的Optional对象为空,或转换函数返回的Optional对象为空,那么最终得到的也是为空的Optional对象。
UserRepository userRepository = new UserRepository();
Optional<User> optionalUser = userRepository.findUserByName("Peter");
Optional<String> optional = optionalUser.flatMap(User::getFullName);
若只需要对Optional对象中的值进行转换,而不需要嵌套的Optional,那么使用map方法更合适
如果要进行一些操作返回另外一个Optional对象,flatmap方法更合适。
6.Stream方法
Optional的stream方法,可以将Optional对象转换为Stream对象,对其中的值进行流操作,如果Optional对象包含值,则将这个值封装到一个Stream流中,如果Optional对象为空,则创造一个为空的Stream流
UserRepository userRepository = new UserRepository();
Optional<User> optionalUser = userRepository.findUserByName("Peter");
Stream<String> a =optionalUser.map(User::getName).stream();
a.forEach(System.out::println);
三、不适合使用Optional对象的情况
不应该用于类的字段,会增加内存消耗,并使序列化变得复杂;
不应该用于方法参数,使方法的理解和使用变得复杂;
不应用于构造器参数,迫使调用者创建Optional实例,应该通过构造器重载解决;
不应该用于集合的参数,集合已经很好的处理空集合的情况,没必要使用Optional包装集合;
不建议使用get方法,若为null会报错。
总结
以上就是今天要讲的内容,本文简单介绍了Optional对象的使用以及不适合使用Optional对象的情况。
声明
本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。