IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> 通过 CompletableFuture 控制线程间依赖关系 -> 正文阅读

[Java知识库]通过 CompletableFuture 控制线程间依赖关系

一?点睛

Future?接口可以创建出新的线程,并在新线程中执行异步操作。但是,?如果使用?Future?接口创建了多个线程,那么这些线程各自独立执行,不存在任务依赖关系,并且无法控制各个线程的执行步骤。

为了解决这种线程间的依赖关系,从?JDK 1.8?开始,Future?接口提供了一个新的实现类?CompletableFuture。CompletableFuture?不但可以创建出异步执行的线程,还可以控制线程的执行步骤,并且可以监控所有线程的结束时刻。例如,可以使用?CompletableFuture?创建?A?和 B?两个线程,然后规定?A 和?B?线程各需要执行 3个?不同的阶段,并且当?A和?B?全部执行完毕后,再触发某个方法。

public class CompletableFuture<T> implements Future<T>, CompletionStage<T> {
    public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier) {
        return asyncSupplyStage(asyncPool, supplier);
    }

    public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier,
                                                   Executor executor) {
        return asyncSupplyStage(screenExecutor(executor), supplier);
    }

    public static CompletableFuture<Void> runAsync(Runnable runnable) {
        return asyncRunStage(asyncPool, runnable);
    }

    public static CompletableFuture<Void> runAsync(Runnable runnable,
                                               Executor executor) {
        return asyncRunStage(screenExecutor(executor), runnable);
    }
}

其中,supplyAsync()?用于有返回值的任务,runAsync()?用于没有返回值的任务,

除了这些用于执行线程任务的方法外,CompletableFuture?还提供了多线程执行时的两种结束逻辑:allOf()?和?anyOf(),如下所述。

1?allOf()?方法会一直阻塞,直到线程池?Executor?中所有线程全部执行完毕。

2?anyOf()?方法会一直阻塞,直到线程池?Executor?中任何一个线程执行完毕。

二?实战

1?需求

有1、2、3、4?四个数字,现要求创建4个线程对这些数字进行处理,并要求每个线程必须按照以下步骤执行。

a?对4个数字的值各加上64,使之转为?A、B、C、D?对应的?ASCII(A:65、B:66、C:67、D:68)。

b?将?ASCII?值转为对应的?A、B、C、D?字符。

c?将转后的?A、B、C、D?加入一个集合中。

最后,还要求当?A、B、C、D 4个线程全部执行完毕后,打印出转换后的结果集。

2?代码

package concurrent;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class CompletableFutureDemo {
    public static void main(String[] args) {
        // 原始数据集
        CopyOnWriteArrayList<Integer> taskList = new CopyOnWriteArrayList();
        taskList.add(1);
        taskList.add(2);
        taskList.add(3);
        taskList.add(4);

        // 结果集
        List<Character> resultList = new ArrayList<>();
        //线程池,可容纳四个线程
        ExecutorService executorService = Executors.newFixedThreadPool(4);

        CompletableFuture[] cfs = taskList.stream()
                // 第一阶段
                .map(integer -> CompletableFuture.supplyAsync(
                        () -> calcASCII(integer), executorService)
                        // 第二阶段
                        .thenApply(i -> {
                            char c = (char) (i.intValue());
                            System.out.println("【阶段2】线程"
                                    + Thread.currentThread().getName() + "执行完毕,"
                                    + "已将int"
                                    + i + "转为了字符" + c);
                            return c;
                        })
                        // 第三阶段
                        .whenComplete((ch, e) -> {
                            resultList.add(ch);
                            System.out.println("【阶段3】线程" +
                                    Thread.currentThread().getName() + "执行完毕," + "已将"
                                    + ch + "增加到了结果集" + resultList + "中");
                            executorService.shutdown();
                        })
                ).toArray(CompletableFuture[]::new);
        // 封装后无返回值,必须自己whenComplete()获取
        CompletableFuture.allOf(cfs).join();//future.get()

        System.out.println("完成!result=" + resultList);
    }

    // 计算i的ASCII值
    public static Integer calcASCII(Integer i) {
        try {
            if (i == 1) {
                Thread.sleep(5000);
            } else {
                Thread.sleep(1000);
            }
            //数字 -> A-D对应的ascii
            i = i + 64;
            System.out.println("【阶段1】线程" + Thread.currentThread().getName()
                    + "执行完毕," + "已将" + i
                    + "转为了A(或B或C或D)对应的ASCII" + i);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return i;
    }
}

3?测试结果

【阶段1】线程pool-1-thread-3执行完毕,已将67转为了A(或B或C或D)对应的ASCII67

【阶段1】线程pool-1-thread-4执行完毕,已将68转为了A(或B或C或D)对应的ASCII68

【阶段1】线程pool-1-thread-2执行完毕,已将66转为了A(或B或C或D)对应的ASCII66

【阶段2】线程pool-1-thread-3执行完毕,已将int67转为了字符C

【阶段2】线程pool-1-thread-4执行完毕,已将int68转为了字符D

【阶段2】线程pool-1-thread-2执行完毕,已将int66转为了字符B

【阶段3】线程pool-1-thread-3执行完毕,已将C增加到了结果集[C, D]中

【阶段3】线程pool-1-thread-4执行完毕,已将D增加到了结果集[C, D]中

【阶段3】线程pool-1-thread-2执行完毕,已将B增加到了结果集[C, D, B]中

【阶段1】线程pool-1-thread-1执行完毕,已将65转为了A(或B或C或D)对应的ASCII65

【阶段2】线程pool-1-thread-1执行完毕,已将int65转为了字符A

【阶段3】线程pool-1-thread-1执行完毕,已将A增加到了结果集[C, D, B, A]中

完成!result=[C, D, B, A]

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-05-09 12:26:14  更:2022-05-09 12:27:09 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 22:51:05-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码