Java并发编程—CompletableFuture的介绍和使用

news2024/12/23 19:22:56

        在博主上一篇博客介绍中,Java并发编程—java异步Future的迭代过程_小魏快起床的博客-CSDN博客,这里面给大家分析了Future的使用过程和一些存在的问题,那么针对里面出现的阻塞问题,博主将在这一篇文章给大家介绍清楚

🍏一、认识新的类CompletableFuture

        🍄1、CompletableFuture类在java源代码中的样子:

public class CompletableFuture<T> implements Future<T>, CompletionStage<T> {	}

        🍄2、CompletableFuture类的架构图

        🍄3、再回顾一下博主上一篇博客中讲的FutureTask类结构图:

        🚗 综上:这个新的CompletableFuture类实现了和FutureTask一样的Future类,所以CompletableFuture类里面有FutureTask关于异步任务的这些特性。同时,CompletableFuture还实现了CompletionStage接口,而真正解决之前的阻塞问题的就是咱们CompletionStage接口,而CompletableFuture进行了一个实现,所以就给大家讲解的是CompletableFuture类;

🍏 二、CompletableFuture和CompletionStage源码分别介绍

        🍄 1、接口CompletionStage:

①、CompletionStage代表异步计算过程中的某一个阶段,一个阶段完成以后可能会触发另外一个阶段。

②、一个阶段的计算执行可以是一个Function,Consumer或者Runnable。

③、一个阶段的执行可能是被单个阶段的完成触发,也可能是由多个阶段一起触发;

        🚗 总结:代表异步计算过程中的某一个阶段,一个阶段完成以后可能会触发另外一个阶段,类似Linux系统的管道分隔符传参数,说人话就是一阶段完成了,需要做二阶段,这个时候二阶段可能会依靠一阶段的结果来执行一些业务。        

        🍄 2、类CompletableFuture:

        CompletableFuture是对Future的扩展和增强。CompletableFuture实现了Future接口,并在此基础上进行了丰富的扩展,完美弥补了Future的局限性,同时CompletableFuture实现了对任务编排的能力。借助这项能力,可以轻松地组织不同任务的运行顺序、规则以及方式。从某种程度上说,这项能力是它的核心能力。而在以往,虽然通过CountDownLatch等工具类也可以实现任务的编排,但需要复杂的逻辑处理,不仅耗费精力且难以维护。

        🚗 总结:CompletableFuture牛逼就完事儿,FutureTask可以做的这个都可以做,并且还可以做一些Future不能做的事;

🍏 三、CompletableFuture的方法介绍

        介绍一下CompletableFuture里面的方法:

runAsync 创建无返回值的异步任务
supplyAsync 创建有返回值的异步任务

代码案例:

public static void main(String[] args) throws ExecutionException, InterruptedException {

        // 创建线程池。线程池的"最大池大小"和"核心池大小"都为1(THREADS_SIZE),"线程池"的阻塞队列容量为1(CAPACITY)。
        ThreadPoolExecutor pool = new ThreadPoolExecutor(
                1,
                1,
                0,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<Runnable>(1)
        );

        //第一个基本方法runAsync
        CompletableFuture<Void> async1 = CompletableFuture.runAsync(() -> {
            System.out.println(Thread.currentThread().getName() + "\t" + "执行");
        });

        //第二个基本方法runAsync,带线程池的
        CompletableFuture<Void> async2 = CompletableFuture.runAsync(() -> {
            System.out.println(Thread.currentThread().getName() + "\t" + "执行");
        }, pool);

        //第三个基本方法supplyAsync
        CompletableFuture<Integer> async3 = CompletableFuture.supplyAsync(() -> {
            System.out.println(Thread.currentThread().getName() + "\t" + "执行");
            return 11;
        });
        System.out.println(async3.get());

        //第四个基本方法supplyAsync
        CompletableFuture<Integer> async4 = CompletableFuture.supplyAsync(() -> {
            System.out.println(Thread.currentThread().getName() + "\t" + "执行");
            return 12;
        }, pool);
        System.out.println(async4.get());

        //关闭线程池
        pool.shutdown();
    }

        🚗 总结:在上面的代码中,每个方法都有2种使用方法,无非就是携带线程池的区别,如果不提供自定义线程池的话,方法内部会用默认的线程池;

         分别执行的效果如下,有数字的是有返回值的,工作中用哪种,取决于实际应用场景:

以上是对这个类和一些方法的初步解析

🍏 四、用CompletableFuompleture做之前的事儿

         之前博主的想法是在主线程中开启一个子线程,然后子线程进行一个计算,然后返回结果值,那么用CompletableFuompleture来做一些试试:

//1、创建一个有返回值的异步任务
        CompletableFuture<Integer> async = CompletableFuture.supplyAsync(() -> {
            try {
                //延迟2s
                TimeUnit.SECONDS.sleep(2);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return 11;
        });
        System.out.println("主线程正常执行");
        //获取异步任务的返回值
        System.out.println(async.get());
        System.out.println("结束");

效果如下:

        分析:这样执行是没啥问题的,但是还是出现了阻塞的问题,明明都说了换类,都说了这个好用,为啥还会出现阻塞呢,这个显然是我们使用错误了,接下来就给大家演示正确的使用方法

 

🍏 五、CompletableFuompleture正确的使用方法:

         博主想做这么一件事儿,开启一个异步任务,第一步计算1+2的值,然后将这个值再提供给下一步做+3的操作,然后输出最后的结果,实现一个多线程异步任务编排:

public static void main(String[] args) throws Exception {
        //1、创建一个线程池
        ThreadPoolExecutor pool = new ThreadPoolExecutor(
                1,
                1,
                0,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<Runnable>(1)
        );

        //2、创建异步编排任务
        CompletableFuture.supplyAsync(() -> {
            //延迟2s
            try { TimeUnit.SECONDS.sleep(2); } catch (InterruptedException e) { }
            //做一下加法然后返回结果
            return 1 + 2;
        }, pool).thenApply(e -> {
            //根据上一步的结果+3
            return e + 3;
        }).whenComplete((v, f) -> {
            //判断是否有异常,如果没有就输出结果值
            if (f == null) {
                System.out.println("结果值:" + v);
            }
        }).exceptionally(e -> {
            //异常输出,whenComplete里面的f不为null时触发
            e.printStackTrace();
            return null;
        });

        //3、主线程执行输出
        System.out.println("输出main线程");

        //4、关闭线程池
        pool.shutdown();
        }

效果如下:

        🚗 总结:在这个案例中,没有用get()方式,用了一些CompletableFuture类里面的自带方法,这就是真正实现了一个异步任务编码的实现,博主这里只用了很简单的例子,铁子们在工作中,完全可以用这种方式来提高效率,这个可以用在数据统计中;

 

🍏 六、常用方法介绍

         在上面的案例中用到了一些方法,在这里可以做一个介绍:      

🍄 1、thenApply / thenApplyAsync 异步回调

前者是由执行job1的线程立即执行job2,即两个job都是同一个线程执行的
后者是将job2提交到线程池中异步执行,实际执行job2的线程可能是另外一个线程

🍄 2、exceptionally whenComplete handle

whenComplete 未发生异常正常返回值发生异常就返回异常,一般配合exceptionally使用
exceptionally确认会发生异常
handle 基本一致 区别在于 handle 有返回值

🍄 3、thenAccept / thenRun

thenAccept 同 thenApply 接收上一个任务的返回值作为参数,但是无返回值;
thenRun 的方法没有入参,也没有返回值

🍕:博主写博客主要是分享技术,分享技术的一个迭代过程,分享技术中发生的问题如何进行解决,欢迎各位铁子留言讨论;

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

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

相关文章

MyBatis框架简介

MyBatis是一个开源的数据持久层框架&#xff0c;内部封装了通过JDBC访问数据库的操作&#xff0c;支持普通的SQL查询、存储过程和高级映射。作为持久层框架&#xff0c;主要思想是将程序中的大量的SQL语句分离出来&#xff0c;配置在相应的配置文件中&#xff0c;这样可以在不修…

Java—数据类型

文章目录数据类型八大基本数据类型Java中有了基本数据类型&#xff0c;为什么还要包装类型String字符串类型函数字符串类的length()方式是否能够得到字符串内有多少个字符&#xff1f;不可变字符串String为什么要设计成不可变的&#xff1f;boolean类型占多少位&#xff1f;为什…

【springboot进阶】使用aop + 注解方式,简单实现spring cache功能

目录 一、实现思路 二、定义缓存注解 三、aop 切面处理 四、使用方式 五、灵活的运用 六、总结 前几天有同学看了 SpringBoot整合RedisTemplate配置多个redis库 这篇文章&#xff0c;提问spring cache 能不能也动态配置多个redis库。介于笔者没怎么接触过&#xff0c;所以…

【Java开发】 Spring 08 :访问 Web 资源( 借助 RestTemplate or WebClient )

web 资源就是运行在服务器上的资源&#xff0c;比如放到 web 下的页面 js 文件、图片、css等&#xff0c;web资源分为静态web资源和动态web资源两类&#xff0c;接下来访问的就是动态资源&#xff08;页面返回的数据是动态的&#xff0c;由后端程序产生&#xff09;&#xff0…

Android 使用元数据

Android 使用元数据 前提介绍Metadata 有时候为安全起见&#xff0c;某个参数要给某个活动专用&#xff0c;并不希望其他活动也能获取该参数&#xff0c;也就是要使用第三方SDK时。Activity提供了元数据&#xff08;Metadata&#xff09;的概念&#xff0c;元数据是一种描述其…

C++类和对象(二)构造函数、析构函数、拷贝构造函数

目录 1.类的6个默认成员函数 2. 构造函数 2.1 概念 2.2 特性 3.析构函数 3.1 概念 3.2 特性 4. 拷贝构造函数 4.1 概念 4.2 特征 1.类的6个默认成员函数 如果一个类中什么成员都没有&#xff0c;简称为空类。 空类中真的什么都没有吗&#xff1f;并不是&#xff0c;…

【菜菜的sklearn课堂笔记】聚类算法Kmeans-聚类算法的模型评估指标

视频作者&#xff1a;菜菜TsaiTsai 链接&#xff1a;【技术干货】菜菜的机器学习sklearn【全85集】Python进阶_哔哩哔哩_bilibili 可以只看轮廓系数和卡林斯基-哈拉巴斯指数 不同于分类模型和回归&#xff0c;聚类算法的模型评估不是一件简单的事。在分类中&#xff0c;有直接结…

【尚硅谷】Java数据结构与算法笔记02 - 队列

文章目录一、使用场景二、队列介绍三、数组模拟队列3.1 思路分析3.2 Java代码实现3.3 问题分析与优化四、数组模拟环形队列4.1 思路分析4.2 Java代码实现一、使用场景 银行排队&#xff0c;先到先得测核酸&#xff0c;先到先测 二、队列介绍 队列是一个有序列表, 可以用数组…

硬盘压缩将C盘拓展成D盘和E盘

硬盘压缩将C盘拓展成D盘和E盘1. 现状2. 硬盘压缩2.1 进入计算机管理2.2 磁盘管理压缩卷3. 分配新盘符3.1 查看盘符是否被占用3.2 新建D盘刚安装好系统的电脑有可能只有一个C盘&#xff0c;我们工作学习的时候远远不够&#xff0c;那怎么拓展其他盘符呢&#xff1f; 接下来让我们…

PyQt5基础练习1

0. 本文学习地址 1. PyQt5是由一系列Python模块组成 超过620个类&#xff0c;6000函数和方法。能在诸如Unix、Windows和Mac OS等主流操作系统上运行。 1.1 PyQt5有两种证书 GPL商业证书 2. 实验1 实现简单的窗体 2.1 完整代码 #!/usr/bin/python3 # -*- coding: utf-8 -*…

专业尖端远心光学,高精度视觉检测解决者

随着机器视觉系统在精密检测领域的广泛应用&#xff0c;在精密光学测量系统中&#xff0c;由于普通光学镜头会存在一定的制约因素&#xff0c;如影像的变形、视角选择而造成的误差、不适当光源干扰下造成边界的不确定性等问题&#xff0c;进而影响测量的精度。为弥补普通镜头应…

155. RESTframe的请求和响应

1.请求和响应 REST framework引入了2个新的对象&#xff1a;Request和Response 1.1 Request 包结构&#xff1a;rest_framework.request.Request 该对象扩展了常规的HttpRequest &#xff0c;增加了对REST框架灵活的请求解析和请求认证的支持 主要属性&#xff1a; data 这个…

基于PHP+MySQL毕业生档案管理系统

毕业生档案管理系统是信息时代的产物&#xff0c;它是学校档案管理部门的一个好帮手。有了它不再需要繁重的纸质登记&#xff0c;有了它档案管理员不在需要繁重的工作&#xff0c;一些成绩信息和奖惩等基本信息可以由管理人员及时的对信息进行查询、更新、修改和删除&#xff0…

SpringBoot_启动原理分析

一共分为三部分来解析: 一 依赖导入原理 二 springboot 包扫描原理三 springboot自动配置原理一 依赖导入原理 父项目进行版本控制 ctrl 点击spring-boot-starter-parent 进入 继续点击,进入spring-boot-dependencies 这里管理着springboot中所有依赖的版本,版本…

38 | Linux 磁盘空间异常爆满

1 场景 收到告警 找到对应的服务器&#xff1a;df -hl 要找到导致磁盘空间满的目录或文件。 2 找占用空间大的目录或文件 2.1方式一 在根目录下&#xff0c;通过du -hs命令&#xff0c;列出各目录所占空间大小 命令&#xff1a;du -hs * 之后再用同样的方法继续到对应目…

[附源码]计算机毕业设计springboot医学图像管理平台

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

EXSI-NFS实验

A-EXSI-NFS实验 2022年3月25日 8:20 iptables -F iptables-save systemctl stop firewalld #setenforce [0|1]命令修改SELinux当前的运行模式&#xff08;0为禁用&#xff0c;1为启用&#xff09; getenforce Enforcing setenforce 0 getenforce Permissive NFS 两台CentOS 7即…

【车载开发系列】UDS诊断---读取内存($0x23)

【车载开发系列】UDS诊断—读取内存&#xff08;$0x23&#xff09; UDS诊断---读取内存&#xff08;$0x23&#xff09;【车载开发系列】UDS诊断---读取内存&#xff08;$0x23&#xff09;一.概念定义二.注意事项三.报文格式1&#xff09;报文请求2&#xff09;肯定响应3&#x…

一、Node.js 环境安装 (详)

1. 下载Node.js 首先进入node.js官网&#xff0c;选择下载这一项&#xff0c;此时映入眼帘的可以看到有两项LTS(长期维护版本)和Current(最新的版本)&#xff0c;一般在开发会选择左边进行下载安装&#xff0c;具稳定性以及有长期维护。那么下面演示的是64位的Windows操作系统&…

PCI bar 解析

只要是接入系统的 pci 设备就需要和系统软件进行交互&#xff0c;设备和系统之间的交流主要包含以下两部分&#xff1a; 1&#xff0c;系统要能访问到设备的寄存器 用于控制设备行为&#xff0c;包括DMA&#xff0c;数据收发等&#xff1b;设备通过寄存器报告自身的状态&…