多线程CompletableFuture之常用方法示例

2023-05-16

文章目录

    • 前期准备
    • 1.runAsync
    • 2.supplyAsync
    • 3.thenRunAsync
    • 4.thenAcceptAsync
    • 5.runAfterBothAsync
    • 6.thenCombineAsync
    • 7.exceptionally
    • 8.handle
    • 9.allOf
    • 10.anyOf

        多线程开发中CompletableFuture必不可少,对比传统的Thread、ThreadPool,CompletableFuture最大的优势是其非常强大的Future的扩展功能,可以在异步方法中获取返回值,类似前端的Promise,CompletableFuture最常用的方法中,run…方法都是没有返回结果的,supply…方法可以获取返回结果。想要写出高并发的程序需要CompletableFuture的支持,CompletableFuture可以完美解决各类复杂的高并发场景。

前期准备

配置ThreadPoolExecutor线程池,在相关类中把线程池注入进来

@Configuration
public class ThreadPoolExecutorConfig {
    @Bean
    public ThreadPoolExecutor threadPoolExecutor() {
        ThreadPoolExecutor executor = new ThreadPoolExecutor(10, 20, 5, TimeUnit.SECONDS, new ArrayBlockingQueue<>(50));
        executor.allowCoreThreadTimeOut(true);
        return executor;
    }
}
	@Autowired
    private ThreadPoolExecutor executor;

1.runAsync

无返回值异步线程,泛型中为void类型,默认主程序不等待子线程

	@GetMapping("/test1")
    public void test1(){
        CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
            try {
                Thread.sleep(200);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务执行结束");
        }, executor);
        System.out.println("执行主程序结束");
    }

执行结果

执行主程序结束
子线程任务执行结束

2.supplyAsync

有返回值异步线程,线程中必须要返回数据,泛型中的值与线程中的返回值需一致,默认主程序不等待子线程。如果需要获取子线程的返回值,调用get方法,主程序会等待子线程结束后执行

	@GetMapping("/test2")
    public void test2() throws ExecutionException, InterruptedException {
        CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(200);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务执行结束");
            return "123";
        }, executor);
        System.out.println("执行主程序结束,不考虑返回值");
        System.out.println("执行主程序结束,线程返回值:"+future.get());
    }

执行结果

执行主程序结束,不考虑返回值
子线程任务执行结束
执行主程序结束,线程返回值:123

3.thenRunAsync

上一步之后链式调用,一般与runAsync搭配使用,thenRunAsync中的方法在runAsync之后执行

	@GetMapping("/test3")
    public void test3(){
        CompletableFuture<Void> future = CompletableFuture.runAsync(() -> {
            try {
                Thread.sleep(200);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务1执行结束");
        }, executor).thenRunAsync(() -> {
            System.out.println("子线程任务2执行结束");
        });
        System.out.println("执行主程序结束");
    }

执行结果

执行主程序结束
子线程任务1执行结束
子线程任务2执行结束

4.thenAcceptAsync

上一步之后调用(可以获取上一步的返回值),一般与supplyAsync搭配使用,thenAcceptAsync中的方法可以在supplyAsync之后才执行,可以获取supplyAsync的返回值

	@GetMapping("/test4")
    public void test4(){
        CompletableFuture<String> future1 = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(200);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务1执行结束");
            return "456";
        }, executor);
        CompletableFuture<Void> future2 = future1.thenAcceptAsync((s) -> {
            System.out.println("子线程任务2获取任务1的数据:" + s);
        }, executor);
        System.out.println("执行主程序结束");
    }

执行结果

执行主程序结束
子线程任务1执行结束
子线程任务2获取任务1的数据:456

5.runAfterBothAsync

任务一、二都完成后执行当前任务,二者都要完成,组合任务不获取前两个任务返回值,且自己无返回值

	@GetMapping("/test5")
    public void test5() {
        CompletableFuture<Integer> future1 = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(200);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务1执行结束");
            return 10 / 2;
        }, executor);
        CompletableFuture<String> future2 = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(300);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务2执行结束");
            return "hello";
        }, executor);
        CompletableFuture<Void> future3 = future1.runAfterBothAsync(future2, () -> {
            System.out.println("子线程任务3执行结束");
        }, executor);
        System.out.println("执行主程序结束");
    }

执行结果

执行主程序结束
子线程任务1执行结束
子线程任务2执行结束
子线程任务3执行结束

6.thenCombineAsync

任务一、二都完成后执行当前任务(可以获取任务一、二的返回值),二者都要完成,组合任务获取前两个任务返回值,且自己有返回值

	@GetMapping("/test6")
    public void test6() throws ExecutionException, InterruptedException {
        CompletableFuture<Integer> future1 = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(200);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务1执行结束");
            return 10 / 2;
        }, executor);
        CompletableFuture<String> future2 = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(300);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务2执行结束");
            return "hello";
        }, executor);
        CompletableFuture<String> future3 = future1.thenCombineAsync(future2, (s1,s2) -> {
            System.out.println("子线程任务3执行结束");
            return s1 + s2;
        }, executor);
        System.out.println("执行主程序结束,不考虑返回值");
        System.out.println("执行主程序结束,线程返回值:future1 = "+future1.get());
        System.out.println("执行主程序结束,线程返回值:future2 = "+future2.get());
        System.out.println("执行主程序结束,线程返回值:future3 = "+future3.get());
    }

执行结果

执行主程序结束,不考虑返回值
子线程任务1执行结束
执行主程序结束,线程返回值:future1 = 5
子线程任务2执行结束
执行主程序结束,线程返回值:future2 = hello
子线程任务3执行结束
执行主程序结束,线程返回值:future3 = 5hello

7.exceptionally

获取上一步的异常,指定某个任务执行异常时执行的回调方法,会将抛出异常作为参数传递到回调方法中

	@GetMapping("/test7")
    public void test7() throws ExecutionException, InterruptedException {
        CompletableFuture<Integer> future1 = CompletableFuture.supplyAsync(()->{
            return 10 / 0;
        }, executor);
        CompletableFuture<Integer> future2 = future1.exceptionally(exception -> {
            System.out.println("出现异常:" + exception);
            return 10; //出现异常,使用默认返回值
        });
        System.out.println("执行主程序结束,线程返回值:"+future2.get());
    }

执行结果

出现异常:java.util.concurrent.CompletionException: java.lang.ArithmeticException: / by zero
执行主程序结束,线程返回值:10

8.handle

方法执行完成后的处理,获取返回值,处理异常,与exceptionally类似

	@GetMapping("/test8")
    public void test8() throws ExecutionException, InterruptedException {
        CompletableFuture<Integer> future1 = CompletableFuture.supplyAsync(()->{
            return 10 / 0;
        }, executor);
        CompletableFuture<Integer> future2 = future1.handle((result,exception) -> {
            if (exception == null) {
                return result;
            }
            System.out.println("handle处理异常:" + exception);
            return 10; //出现异常,使用默认返回值
        });
        System.out.println("执行主程序结束,线程返回值:"+future2.get());
    }

执行结果

handle处理异常:java.util.concurrent.CompletionException: java.lang.ArithmeticException: / by zero
执行主程序结束,线程返回值:10

9.allOf

阻塞等待所有异步任务返回

	@GetMapping("/test9")
    public void test9() throws ExecutionException, InterruptedException {
        CompletableFuture<Void> future1 = CompletableFuture.runAsync(()->{
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务1执行结束");
        }, executor);
        CompletableFuture<Void> future2 = CompletableFuture.runAsync(()->{
            try {
                Thread.sleep(200);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务2执行结束");
        }, executor);
        CompletableFuture<Void> future3 = CompletableFuture.runAsync(()->{
            try {
                Thread.sleep(300);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务3执行结束");
        }, executor);
        // 阻塞等待所有异步任务返回
        CompletableFuture.allOf(future1, future2, future3).get();
        System.out.println("执行主程序结束");
    }

执行结果

子线程任务1执行结束
子线程任务2执行结束
子线程任务3执行结束
执行主程序结束

10.anyOf

所有异步任务只要有一个完成

	@GetMapping("/test10")
    public void test10() throws ExecutionException, InterruptedException {
        CompletableFuture<Void> future1 = CompletableFuture.runAsync(()->{
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务1执行结束");
        }, executor);
        CompletableFuture<Void> future2 = CompletableFuture.runAsync(()->{
            try {
                Thread.sleep(200);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务2执行结束");
        }, executor);
        CompletableFuture<Void> future3 = CompletableFuture.runAsync(()->{
            try {
                Thread.sleep(300);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("子线程任务3执行结束");
        }, executor);
        // 阻塞等待某一个异步任务返回
        CompletableFuture.anyOf(future1, future2, future3).get();
        System.out.println("执行主程序结束");
    }

执行结果

子线程任务1执行结束
执行主程序结束
子线程任务2执行结束
子线程任务3执行结束
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

多线程CompletableFuture之常用方法示例 的相关文章

  • python的panda库读写文件

    目录 1 读取excel文件 xff08 1 xff09 语法 xff08 2 xff09 实例 2 读取cvs文件 xff08 1 xff09 语法 xff08 2 xff09 实例 3 读取txt文件 xff08 1 xff09 语法
  • frankmocap

    1 配置和报错 1 1 模块缺失 ModuleNotFoundError No module named detectors body pose estimator pose2d models 下载问题 xff0c 下载完之后该文件夹下面的
  • java的三种代码注释方式

    1 注释的作用 注释是对程序语言的说明 xff0c 有助于开发者之间的交流 xff0c 方便理解和维护程序 注释不是编程语句 xff0c 不会被编译器执行 一些代码量少较少的程序 xff0c 我们加不加注释对我们的理解和修改代码没有太大影响
  • UCOS的多任务

    初学UCOS多任务调度 xff0c 总是有这样一个疑惑 xff1a 为什么多使用多任务 xff0c 写多个函数不就可以了吗 xff1f 随着不断学习 xff0c 后面终于有了一个答案 多任务是并发执行 xff0c 每个任务都做一点 xff1
  • IAR编译成功,但烧录不进去

    把默认的Simulator改为ST LINK试试 xff1f
  • ipmitool使用

    BMC IPMI常用命令 BMC Baseboard Management Controler 提供了多种通道来和主机通信 xff0c 进而检测主机的温度 风扇转速 电压 电源和现场可替代器件 为了便于用户使用 xff0c 它提供了非常丰富
  • 基于LMI的等效滑模控制

    目录 前言 1 一阶欠驱动倒立摆系统 2 基于LMI的等效滑模控制器 3 simulink仿真 3 1 simulink模型 3 2 结果分析 3 2 结论 前言 关于LMI和滑模控制的结合上两篇文章已有介绍和仿真分析 xff0c 本篇文章
  • 基于扩张观测器(LESO)的滑模控制

    目录 前言 1 二阶系统LESO观测器设计 2 基于LESO的滑模控制器设计 3 仿真分析 普通高增益项 3 1仿真模型 3 2仿真结果 3 3 总结 4 仿真分析 优化后的高增益项 4 1 优化高增益项 4 2仿真结果 4 2 1 高增益
  • 基于遗传算法和粒子群算法的PID悬架控制、LQR悬架控制和滑模悬架控制

    目录 1 基于遗传算法和粒子群算法的的PID悬架控制 1 1 两种悬架系统 1 1 1 将路面激励整合到悬架系统 1 1 2 不将路面激励整合到悬架系统 1 1 3 总结 1 2 PID经典控制理论 1 3 优化PID参数的目标函数和约束条
  • 2自由度陀螺仪滑模控制和PID控制跟踪目标轨迹

    目录 前言 1 陀螺仪模型 2 滑模跟踪控制 3 PID控制 4 总结 1 陀螺仪模型 2 滑模跟踪控制 对于2自由度陀螺仪有两个方向x y跟踪 xff0c 所以需要分别为两个方向单独设计滑模面 xff0c 这里仍以简单的线性滑模面设计分析
  • 自抗扰控制ADRC之三种微分跟踪器TD仿真分析

    目录 前言 1 全程快速微分器 1 1仿真分析 1 2仿真模型 1 3仿真结果 1 4结论 2 Levant微分器 2 1仿真分析 2 2仿真模型 2 3仿真结果 3 非线性跟踪微分器 韩教授 3 1仿真分析 3 2小结 4 总结 前言 工
  • 自抗扰控制ADRC之扩张观测器

    目录 前言 1 被控对象 被观测对象 2 非线性观测器 2 1仿真分析 2 2仿真模型 2 3仿真结果 3 线性观测器 3 1仿真模型 3 2仿真结果 4 总结和学习问题 前言 什么叫观测器 xff1f 为什么该类观测称为扩张观测器 xff
  • 基于神经网络(RBF)补偿的双关节机械手臂自适应控制

    目录 前言 1 双关节机械手臂模型 1 1 实际模型 1 2 名义模型 2 控制律设计 3 神经网络补偿自适应律设计 3 1自适应律 3 2自适应律 4 仿真分析 4 1仿真模型 4 2 仿真结果 4 3 小结 5学习问题 前言 所谓的补偿
  • PID、模糊PID、SkyHook、LQR、H2/H∞、ADRC等悬架控制合集

    罗列一下现成的悬架模型以及应用的控制算法 xff1a PID 模糊PID SkyHook LQR H2 H ADRC等 xff0c 以及kalman观测器 xff1a 半车 前后 左右 整车悬架详细推导建模和simulink仿真分析 侧倾
  • 1086:角谷猜想(C C++)

    题目描述 谓角谷猜想 xff0c 是指对于任意一个正整数 xff0c 如果是奇数 xff0c 则乘3加1 xff0c 如果是偶数 xff0c 则除以2 xff0c 得到的结果再按照上述规则重复处理 xff0c 最终总能够得到1 如 xff0
  • 渗透测试工具之Metasploit Framework(MSF)

    一 Metasploit Framework xff08 MSF xff09 简介 Metasploit是当前信息安全与渗透测试领域最流行的术语 xff0c 它的出现颠覆了以往的已有的渗透测试的方式 几乎所有流行的操作系统都支持Metasp
  • Test Case Framework (TCF) 简介

    TCF is a system that simplifies the creation and execution of test cases automation for that matter with minimal setup e
  • 小菜鸡的第一天

    1 CPU分配 xff0c i5四核所以给Linux分配两个核 2 内存分配 xff0c 有16GB所以分配8个g 3 硬盘 xff0c 需要创立一个独立的分区 xff0c 最好300G以上 xff08 由于个人原因分配了60G xff09
  • 小菜鸡的第二天

    绝对路径 xff1a 以 开头 代表当前路径 xff0c 或者 代表上一级目录 xff0c 或者 插入U盘之后 xff0c 在 dev文件夹下输入 ls sd 可以看到U盘信息 xff0c 要想知道插入的U盘是哪个 xff0c 重新插拔再次
  • 小菜鸡的第三天

    压缩与解压 tar vcjf 43 文件名 tar bz2 xxx xxx是要压缩的文件名 压缩命令 tar vxjf 43 xxx tar bz2 xxx是要解压缩的文件名 解压缩命令 f xff1a 使用归档文件或ARCHIVE设备 c

随机推荐