JDK1.8新特性CompletableFuture总结

CompletableFuture这个completableFuture是JDK1.8版本新引入的类。下面是这个类。实现了俩接口。本身是个class。这个是Future的实现类。使用completionStage接口去支持完成时触发的函数和操作。一个completetableFuture就代表了一个任务。他能用Future的方法。还能做一些之前说的executorService配合fu…

大家好,又见面了,我是你们的朋友全栈君。

CompletableFuture

这个completableFuture是JDK1.8版本新引入的类。下面是这个类。实现了俩接口。本身是个class。这个是Future的实现类。

使用completionStage接口去支持完成时触发的函数和操作。

一个completetableFuture就代表了一个任务。他能用Future的方法。还能做一些之前说的executorService配合futures做不了的。

之前future需要等待isDone为true才能知道任务跑完了。或者就是用get方法调用的时候会出现阻塞。而使用completableFuture的使用就可以用then,when等等操作来防止以上的阻塞和轮询isDone的现象出现。

JDK1.8新特性CompletableFuture总结

1. 创建CompletableFuture直接new对象也成。一个completableFuture对象代表着一个任务。这个对象能跟这个任务产生联系。下面用的complete方法意思就是这个任务完成了需要返回的结果。然后用get();方法可以获取到。

JDK1.8新特性CompletableFuture总结

2.JDK1.8使用的接口类。在本文的CompletableFuture中大量的使用了这些函数式接口。

注:这些声明大量应用于方法的入参中。像thenApply和thenAccept这俩就是一个用Function一个用Consumer

而lambda函数正好是可以作为这些接口的实现。例如 s->{return 1;} 这个就相当于一个Function。因为有入参和返回结果。

JDK1.8新特性CompletableFuture总结

(1)Function

JDK1.8新特性CompletableFuture总结

(2)Consumer

JDK1.8新特性CompletableFuture总结

对于前面有Bi的就是这样的。BiConsumer就是两个参数的。

JDK1.8新特性CompletableFuture总结

(3)Predicate这个接口声明是一个入参,返回一个boolean。

JDK1.8新特性CompletableFuture总结

(4)supplier

JDK1.8新特性CompletableFuture总结

3. 下面是这个类的静态方法。带有Async就是异步执行的意思、也是一个completableFuture对象代表着一个任务这个原则。

这种异步方法都可以指定一个线程池作为任务的运行环境。如果没有指定就会使用ForkJoinPool线程池来执行、

JDK1.8新特性CompletableFuture总结

(1)supplyAsync&runAsync的使用例子。

public static void main(String[] args) throws ExecutionException, InterruptedException {
    ExecutorService executorService = Executors.newCachedThreadPool();
    executorService.submit(new Callable<Object>() {
        @Override
        public Object call() throws Exception {
            System.out.println("executorService 是否为守护线程 :" + Thread.currentThread().isDaemon());
            return null;
        }
    });
    final CompletableFuture<String> completableFuture = CompletableFuture.supplyAsync(() -> {
        System.out.println("this is lambda supplyAsync");
        System.out.println("supplyAsync 是否为守护线程 " + Thread.currentThread().isDaemon());
        try {
            TimeUnit.SECONDS.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("this lambda is executed by forkJoinPool");
        return "result1";
    });
    final CompletableFuture<String> future = CompletableFuture.supplyAsync(() -> {
        System.out.println("this is task with executor");
System.out.println("supplyAsync 使用executorService 时是否为守护线程 : " + Thread.currentThread().isDaemon());
        return "result2";
    }, executorService);
    System.out.println(completableFuture.get());
    System.out.println(future.get());
    executorService.shutdown();
}

JDK1.8新特性CompletableFuture总结

注意:这些任务中。带有supply是持有返回值的,run是void返回值的。在玩supply时发现一个问题如果使用supplyAsync任务时不使用任务的返回值。即不用get方法阻塞主线程会导致任务执行中断。(注:跟get方法无关,后面有答案)

JDK1.8新特性CompletableFuture总结

JDK1.8新特性CompletableFuture总结

然后我开始探索是否是只有supplyAsync是这样。我测试了runAsync发现也是这样。

JDK1.8新特性CompletableFuture总结

下图为与supplyAsync任务执行不全面一样的问题。我甚至测试了将lambda换成runnable发现无济于事。

JDK1.8新特性CompletableFuture总结

答案:

造成这个原因是因为Daemon。因为completableFuture这套使用异步任务的操作都是创建成了守护线程。那么我们没有调用get方法不阻塞这个主线程的时候。主线程执行完毕。所有线程执行完毕就会导致一个问题,就是守护线程退出。那么我们没有执行的代码就是因为主线程不再跑任务而关闭导致的。可能这个不叫问题,因为在开发中我们主线程常常是一直开着的。但是这个小问题同样让我想了好久。下面我们开一个非守护线程,可以看到程序执行顺利。

JDK1.8新特性CompletableFuture总结

下面证实守护线程在其他非守护线程全部退出的情况下不继续执行。

final CompletableFuture<String> completableFuture = CompletableFuture.supplyAsync(() -> {
    System.out.println("this is lambda supplyAsync");
    System.out.println("supplyAsync 是否为守护线程 " + Thread.currentThread().isDaemon());
    try {
        TimeUnit.SECONDS.sleep(1);
        try(BufferedWriter writer = new BufferedWriter
                (new OutputStreamWriter(new FileOutputStream(new File("/Users/zhangyong/Desktop/temp/out.txt"))))){
            writer.write("this is completableFuture daemon test");
        }catch (Exception e){
            System.out.println("exception find");
        }
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    System.out.println("this lambda is executed by forkJoinPool");
    return "result1";
});

这个代码就是操作本地文件。并且sleep了一秒。其他线程就一句控制台输出的代码。最终的结果是文件没有任何变化。

当我把主线程sleep 5秒时。本地文件会写入一句  this is completableFuture daemon test   验证成功。

(2)allOf&anyOf

这两个方法的入参是一个completableFuture组、allOf就是所有任务都完成时返回。但是是个Void的返回值。

anyOf是当入参的completableFuture组中有一个任务执行完毕就返回。返回结果是第一个完成的任务的结果。

public static void otherStaticMethod() throws ExecutionException, InterruptedException {
        final CompletableFuture<String> futureOne = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                System.out.println("futureOne InterruptedException");
            }
            return "futureOneResult";
        });
        final CompletableFuture<String> futureTwo = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(6000);
            } catch (InterruptedException e) {
                System.out.println("futureTwo InterruptedException");
            }
            return "futureTwoResult";
        });
        CompletableFuture future = CompletableFuture.allOf(futureOne, futureTwo);
        System.out.println(future.get());
//        CompletableFuture completableFuture = CompletableFuture.anyOf(futureOne, futureTwo);
//        System.out.println(completableFuture.get());
    }

JDK1.8新特性CompletableFuture总结

JDK1.8新特性CompletableFuture总结

(3)completedFuture这个方法我没懂他是干啥的。源码就是返回一个值。感觉没啥意义。

JDK1.8新特性CompletableFuture总结

(4)取值方法,除了get还有一个getNow(); 这个就比较特殊了。 这个方法是执行这个方法的时候任务执行完了就返回任务的结果,如果任务没有执行完就返回你的入参。

JDK1.8新特性CompletableFuture总结

(5)join方法跟线程的join用法差不多。

JDK1.8新特性CompletableFuture总结

(6)whenXXX,在一个任务执行完成之后调用的方法。

这个有三个名差不多的方法。whenComplete、whenCompleteAsync、还有一个是whenCompleteAsync用自定义Executor

JDK1.8新特性CompletableFuture总结

首先看一下这个whenComplete实例方法。这个就是任务执行完毕调用的,传入一个action。这个方法的执行线程是当前线程,意味着会阻塞当前线程。下面图中test的输出跟whenComplete方法运行的线程有关。运行到main线程就会阻塞test的输出。运行的是completableFuture线程则不会阻塞住test的输出。

JDK1.8新特性CompletableFuture总结

下面是任务执行的线程的探索。

JDK1.8新特性CompletableFuture总结

JDK1.8新特性CompletableFuture总结

根据测试得出的结论是:如果调用whenComplete的中途,还发生了其他事情,图中的主线程的sleep(400);导致completableFuture这个任务执行完毕了,那么就使用主线程调用。如果调用的中途没有发生其他任务且在触碰到whenComplete方法时completableFuture这个任务还没有彻底执行完毕那么就会用completableFuture这个任务所使用的线程。

 

下面是whenCompleteAsync方法。这个方法就是新创建一个异步线程执行。所以不会阻塞。

JDK1.8新特性CompletableFuture总结

(7) then。方法瞅着挺多的。实际上就是异不异步和加不加自定义Executor.

JDK1.8新特性CompletableFuture总结

注:whenComplete中出现的问题在then中测试不存在、使用的就是上一个任务的线程。这个thenCompose就是一个任务执行完之后可以用它的返回结果接着执行的方法。方法返回的是另一个你期盼泛型的结果、

compose理解就是上一个任务结果是then的一部分。

JDK1.8新特性CompletableFuture总结

下面介绍一下thenCombine。这个combine的理解就是结合两个任务的结果。

JDK1.8新特性CompletableFuture总结

综上:这个线程的问题并不是大问题,只要你不用线程来做判断条件。他并不会影响你的效率。试想pool线程都执行完了就用主线程跑呗。没跑完,而使你等了那你就用pool线程呗。

thenRun就是这个任务运行完,再运行下一个任务。感觉像是join了一下。

JDK1.8新特性CompletableFuture总结

其余不再介绍。大同小异。

像thenApply(Function);这样的就是有入参有返回值类型的。

像thenAccept(Consumer);这样的就是有入参,但是没有返回值的。详情在上文中有过关于函数式接口的叙述。

 

 

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/124952.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • pycharm怎么创建虚拟环境_pycharm选择运行环境

    pycharm怎么创建虚拟环境_pycharm选择运行环境配置虚拟环境的必要性1、保持项目的独立性。为每一个项目建立一个独立的运行环境,不与其他项目之间产生环境冲突。(比如其他项目加载或删除依赖环境软件时,互不影响)通俗一点就是,每人一个单间,大家在自己的房间内可以各干各的。2、提升项目开机加载速度。在独立性的前提下,虚拟环境可以选择继承已有的公共环境中的依赖包,也可以完全新建一个空间,也可以将自己的依赖环境共享给其他项目。总之,保证每个项目的虚…

    2022年8月28日
    4
  • pycharm2021.3激活码破解方法

    pycharm2021.3激活码破解方法,https://javaforall.net/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

    2022年3月14日
    373
  • linux内核编译过程的最终总结版

    linux内核编译过程的最终总结版一、实验目的学习重新编译Linux内核,理解、掌握Linux内核和发行版本的区别。二、实验内容在Linux操作系统环境下重新编译内核。实验主要内容:A.查找并且下载一份内核源代码,本实验使用最新的Linux内核2.6.36。B.配置内核。C.编译内核和模块。D.配置启动文件。本次实验环境是Linux2.6.35内核的环境下,下载并重新编译内核源代码(2…

    2022年7月23日
    16
  • 人工智能技术应用的领域主要有哪些?

    人工智能技术应用的领域主要有哪些?随着智能家电、穿戴设备、智能机器人等产物的出现和普及,人工智能技术已经进入到生活的各个领域,引发越来越多的关注。那么,人工智能目前都应用在哪些领域,运用了怎样的技术原理呢?  什么是人工智能?  人工智能是研究、开发用于模拟、延伸和扩展人的智能的理论、方法、技术及应用系统的一门新的技术科学,是认知、决策、反馈的过程。    曾经有很多人戏称,人工智能就像一列火车,你苦苦期盼,它终于来了,

    2022年5月4日
    124
  • 让你轻松架设FTP服务器

    让你轻松架设FTP服务器你想架设你的FTP服务器吗?笔者将从IIS和第三方软件两个角度,教会你轻松架设FTP服务器的方法,让你玩转FTP服务器的架设。一、用IIS架设FTP服务器:1、设置FTP服务器:(1)设置“FTP站点”标签项:在“控制面板→管理工具→Internet服务管理器”窗口中,选中“默认FTP站点→右键→属性”,在图1的“默认FTP站点属性”框中,“IP地址”可以使用默认,端口号保持“2

    2022年7月21日
    11
  • 详细总结BS架构和CS架构

    详细总结BS架构和CS架构文章目录CS架构CS结构CS的优缺点BS架构BS结构BS的优缺点CS与BS的对比总结文章内容学习百度百科和博主“向上人生”CS架构客户端-服务器,即Client-Server(C/S)结构。C/S结构通常采取两层结构。服务器负责数据的管理,客户端负责完成与用户的交互任务。CS结构在C/S结构中,应用程序分为两部分:服务器部分和客户端部分。服务器部分是多个用户共享的信息与功能,执行后台服务,如控制共享数据库的操作等;客户端部分为用户所专有,负责执行前台功能,在出错提示、在线帮助等方面都有强大

    2022年6月20日
    55

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号