Java Optional orElse orElseGet orElseThrow()

在 Java 中,Optional orElse类型 包含 .orElse() .orElseThrow()

orElseThrow() 用法:

无参版本:当 Optional 为空时,抛出 NoSuchElementException;

java 复制代码
Optional<String> optionalValue = Optional.empty();
String value = optionalValue.orElseThrow(); // 当为空时抛出 NoSuchElementException

带参版本:可以自定义抛出的异常类型

java 复制代码
Optional<String> optionalValue = Optional.empty();
String value = optionalValue.orElseThrow(() -> new IllegalArgumentException("值不存在"));

用法示例:

java 复制代码
import java.util.Optional;
import java.util.function.Supplier;

public class OrElseGetExample {
    public static void main(String[] args) {
        // 1. Optional 有值的情况
        Optional<String> presentOpt = Optional.of("存在的值");
        String result1 = presentOpt.orElseGet(() -> {
            System.out.println("执行了默认值逻辑(本不该执行)"); // 不会执行
            return "默认值";
        });
        System.out.println(result1); // 输出:存在的值


        // 2. Optional 为空的情况
        Optional<String> emptyOpt = Optional.empty();
        String result2 = emptyOpt.orElseGet(() -> {
            System.out.println("执行了默认值逻辑"); // 会执行
            return "通过Supplier生成的默认值";
        });
        System.out.println(result2); // 输出:通过Supplier生成的默认值


        // 3. 实际场景:复杂默认值的创建(如数据库查询)
        Optional<User> userOpt = Optional.empty(); // 假设从数据库查询结果为空
        User defaultUser = userOpt.orElseGet(() -> fetchDefaultUserFromDB());
        System.out.println("使用的用户:" + defaultUser.getName()); // 输出:默认用户
    }

    // 模拟从数据库获取默认用户(成本较高的操作)
    private static User fetchDefaultUserFromDB() {
        System.out.println("执行数据库查询获取默认用户...");
        return new User("默认用户");
    }

    static class User {
        private String name;
        public User(String name) { this.name = name; }
        public String getName() { return name; }
    }
}

使用时要注意,如果 Optional 为空,这个方法会立即抛出异常,而不是返回默认值,这是它与 orElse()orElseGet() 的主要区别。

orElseGet() 用法:

java 复制代码
import java.util.Optional;
import java.util.function.Supplier;

public class OrElseGetExample {
    public static void main(String[] args) {
        // 1. Optional 有值的情况
        Optional<String> presentOpt = Optional.of("存在的值");
        String result1 = presentOpt.orElseGet(() -> {
            System.out.println("执行了默认值逻辑(本不该执行)"); // 不会执行
            return "默认值";
        });
        System.out.println(result1); // 输出:存在的值


        // 2. Optional 为空的情况
        Optional<String> emptyOpt = Optional.empty();
        String result2 = emptyOpt.orElseGet(() -> {
            System.out.println("执行了默认值逻辑"); // 会执行
            return "通过Supplier生成的默认值";
        });
        System.out.println(result2); // 输出:通过Supplier生成的默认值


        // 3. 实际场景:复杂默认值的创建(如数据库查询)
        Optional<User> userOpt = Optional.empty(); // 假设从数据库查询结果为空
        User defaultUser = userOpt.orElseGet(() -> fetchDefaultUserFromDB());
        System.out.println("使用的用户:" + defaultUser.getName()); // 输出:默认用户
    }

    // 模拟从数据库获取默认用户(成本较高的操作)
    private static User fetchDefaultUserFromDB() {
        System.out.println("执行数据库查询获取默认用户...");
        return new User("默认用户");
    }

    static class User {
        private String name;
        public User(String name) { this.name = name; }
        public String getName() { return name; }
    }
}

orElse() 用法:

java 复制代码
import java.util.Optional;

public class OrElseExample {
    public static void main(String[] args) {
        // 1. Optional 有值的情况
        Optional<String> presentOpt = Optional.of("存在的值");
        String result1 = presentOpt.orElse("默认值");
        System.out.println(result1); // 输出:存在的值


        // 2. Optional 为空的情况
        Optional<String> emptyOpt = Optional.empty();
        String result2 = emptyOpt.orElse("默认值");
        System.out.println(result2); // 输出:默认值


        // 3. 注意:默认值的逻辑总会执行(即使 Optional 有值)
        Optional<String> opt = Optional.of("有值");
        String result3 = opt.orElse(generateDefaultValue()); 
        // 这里会先执行 generateDefaultValue(),再返回 "有值"
    }

    // 生成默认值的方法
    private static String generateDefaultValue() {
        System.out.println("执行了生成默认值的逻辑");
        return "动态生成的默认值";
    }
}

实际开发中需要按具体需求去使用它们

相关推荐
小bo波2 小时前
使用Thread子类创建线程 VS 使用Runnable接口创建线程的区别
java·多线程·thread·并发编程·runnable
SamDeepThinking2 小时前
高并发场景下,CompletableFuture与ForkJoinPool该如何取舍?
java·后端·面试
张不才5 小时前
CPU 100% 了怎么办?Java 性能排障的标准化操作
java·后端
shepherd1117 小时前
吞吐量提升 10 倍:高并发大批量数据处理任务的架构演进与性能调优
java·后端·架构
plainGeekDev9 小时前
单例模式 → object 声明
android·java·kotlin
用户2986985301410 小时前
Java 实现 Word 文档文本与图片提取的方法
java·后端
SimonKing11 小时前
铁子,IntelliJ IDEA 2026.1.3来了,升不升?
java·后端·程序员
咖啡八杯1 天前
GoF设计模式——策略模式
java·后端·spring·设计模式
用户128526116021 天前
我把祖传Java项目重构后,接口响应从3s砍到了200ms,只改了这几行代码
java