首页 > 编程语言 >JUC并发编程(1)—CompletableFuture详解

JUC并发编程(1)—CompletableFuture详解

时间:2023-07-19 20:45:06浏览次数:54  
标签:JUC System 详解 线程 println CompletableFuture public out

@

目录

最近在学习juc并发编程,于是决定汇总一下并发编程中常用方法,常见问题以及常见考题,今天是第一章—CompletableFuture

CompletableFuture介绍

CompletableFuture是jdk8版本开始出现的类。目的是为了应用于并发编程状态下遇到的各种场景。CompletableFuture实现了CompletionStage接口和Future接口,对Java7及以前Future接口做了大量的扩展,增加了许多常用方法,增加了异步会点、流式处理、多个Future组合处理的能力,使Java在处理多任务的协同工作时更加顺畅便利。

首先得学会看懂几个函数式接口的特性!!!(重点)
在这里插入图片描述

1.创建异步任务

//runAsync方法不支持返回值
public static CompletableFuture<Void> runAsync(Runnable runnable)
public static CompletableFuture<Void> runAsync(Runnable runnable, Executor executor)
//supplyAsync可以支持返回值
public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier)
public static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier, Executor executor)

CompletableFuture创建异步任务主要分为两个方法runAsync和supplyAsync
其中

  1. runAsync方法不支持返回值
  2. supplyAsync可以支持返回值(常用)
  3. 没有指定Executor的方法会使用ForkJoinPool.commonPool() 作为它的线程池执行异步代码。如果指定线程池,则使用指定的线程池运行。

下面是四种创建方式

public class CompletableFutureTest {
   public static void main(String[] args) throws Exception{
       ThreadPoolExecutor executor = new ThreadPoolExecutor(
      			5,5,5
               TimeUnit.MINUTES,
               new LinkedBlockingQueue<>(10));
       CompletableFuture future1=CompletableFuture.runAsync(()->{
       System.out.println(Thread.currentThread().getName()+"*********future1 coming in");
       });
       //这里获取到的值是null,无返回值
       System.out.println(future1.get());
       CompletableFuture<Void> future2 = CompletableFuture.runAsync(() -> {
           //ForkJoinPool.commonPool-worker-9 
           System.out.println(Thread.currentThread().getName() + "\t" + "*********future2 coming in");
       }, executor);
       CompletableFuture<Integer> future3 =CompletableFuture.supplyAsync(()-> {
           //pool-1-thread-1
           System.out.println(Thread.currentThread().getName() + "\t" + "future3带有返回值");
           return "abc";
       });
       System.out.println(future3.get());
       CompletableFuture<Integer> future4 = CompletableFuture.supplyAsync(() -> {
           System.out.println(Thread.currentThread().getName() + "\t" + "future4带有返回值");
           return "abc";
       }, executor);
       System.out.println(future4.get());
       //关闭线程池
       executor.shutdown();
   }
}

2.CompletableFuture API

①. 获得结果和触发计算(get、getNow、join、complete)

获得结果和触发计算(get、getNow、join、complete)

  1. public T get( ):不见不散(会抛出异常) 只要调用了get( )方法,不管是否计算完成都会导致阻塞
  2. public T get(long timeout, TimeUnit unit):过时不候
  3. public T getNow(T valuelfAbsent):没有计算完成的情况下,给我一个替代结果计算完,返回计算完成后的结果、没算完,返回设定的valuelfAbsent
  4. public T join( ):join方法和get( )方法作用一样,唯一区别是join方法编译时不需要手动抛异常

在这里插入图片描述

  1. public CompletableFuture thenApply:计算结果存在依赖关系,这两个线程串行化,由于存在依赖关系(当前步错,不走下一步),当前步骤有异常的话就叫停
  2. public CompletableFuture handle(BiFunction<? super T, Throwable, ? extends U> fn):有异常也可以往下一步走,根据带的异常参数可以进一步处理
  3. whenComplete:是执行当前任务的线程执行继续执行whenComplete的任务
  4. whenCompleteAsync:是执行把whenCompleteAsync这个任务继续提交给线程池来进行执行

②. 对计算结果进行处理(thenApply、handle)

  1. public CompletableFuture thenApply:计算结果存在依赖关系,这两个线程串行化由于存在依赖关系(当前步错,不走下一步),当前步骤有异常的话就叫停
  2. public CompletableFuture handle(BiFunction<? super T, Throwable, ? extends U> fn):有异常也可以往下一步走,根据带的异常参数可以进一步处理
  3. whenComplete:是执行当前任务的线程执行继续执行whenComplete的任务
  4. whenCompleteAsync:是执行把whenCompleteAsync这个任务继续提交给线程池来进行执行
    在这里插入图片描述
     CompletableFuture<Integer> future = CompletableFuture.supplyAsync(() -> {
         try { TimeUnit.SECONDS.sleep(1);  } catch (InterruptedException e) {e.printStackTrace();}
         return 1;
     }).thenApply(s->{
         System.out.println("-----1");
         //如果加上int error=1/0; 由于存在依赖关系(当前步错,不走下一步),当前步骤有异常的话就叫停
         //int error=1/0;
         return s+1;
     }).thenApply(s->{
         System.out.println("-----2");
         return s+2;
     }).whenComplete((v,e)->{
         if(e==null){
             System.out.println("result-----"+v);
         }
     }).exceptionally(e->{
         e.printStackTrace();
         return null;
     });
     System.out.println(Thread.currentThread().getName()+"\t"+"over....");
     try { TimeUnit.SECONDS.sleep(3);  } catch (InterruptedException e) {e.printStackTrace();}

		CompletableFuture.supplyAsync(() -> {
            return 1;
        }).handle((f,e) -> {
            System.out.println("-----1");
            return f + 2;
        }).handle((f,e) -> {
            System.out.println("-----2");
            int error=1/0;
            return f + 3;
        }).handle((f,e) -> {
            System.out.println("-----3");
            return f + 4;
        }).whenComplete((v, e) -> {
            if (e == null) {
                System.out.println("----result: " + v);
            }
        }).exceptionally(e -> {
            e.printStackTrace();
            return null;
        }).join());

③. 对计算结果进行消费(thenRun、thenAccept、thenApply)

  1. thenRun(Runnable runnable)
    任务A执行完执行B,并且B不需要A的结果
  2. CompletableFutur thenAccept(Consumer<? super T> action)
    任务A执行完成执行B,B需要A的结果,但是任务B无返回值
  3. public CompletableFuture thenApply(Function<? super T,? extends U> fn)

在这里插入图片描述
4. 线程串行化方法
带了Async的方法表示的是:会重新在线程池中启动一个线程来执行任务

public <U> CompletableFuture<U> thenApply(Function<? super T,? extends U> fn)
public <U> CompletableFuture<U> thenApplyAsync(Function<? super T,? extends U> fn)
public <U> CompletableFuture<U> thenApplyAsync(Function<? super T,? extends U> fn, Executor executor)
public CompletableFuture<Void> thenAccept(Consumer<? super T> action)
public CompletableFuture<Void> thenAcceptAsync(Consumer<? super T> action)
public CompletableFuture<Void> thenAcceptAsync(Consumer<? super T> action,Executor executor)
public CompletableFuture<Void> thenRun(Runnable action)
public CompletableFuture<Void> thenRunAsync(Runnable action)
public CompletableFuture<Void> thenRunAsync(Runnable action,Executor executor)         

④. 对计算速度进行选用(applyToEither、acceptEither、runAfterEither)

  1. public CompletableFuture applyToEither(CompletionStage<? extends T> other, Function<? super T, U> fn)
    这个方法表示的是,谁完成任务完成的快就返回谁的结果
    在这里插入图片描述
  2. 两任务组合,一个完成
    applyToEither:两个任务有一个执行完成,获取它的返回值,处理任务并有新的返回值
    acceptEither:两个任务有一个执行完成,获取它的返回值,处理任务,没有新的返回值
    runAfterEither:两个任务有一个执行完成,不需要获取 future 的结果,处理任务,也没有返回值

⑤. 对计算结果进行合并(thenCombine、thenAcceptBoth、runAfterBoth)

  1. public <U,V> CompletableFuture thenCombine(CompletionStage<? extends U> other,BiFunction<? super T,? super U,? extends V> fn)
    两个CompletionStage任务都完成后,最终把两个任务的结果一起交给thenCombine来处理先完成的先等着,等待其他分支任务
    在这里插入图片描述

⑥. 多任务组合(allOf、anyOf)

  1. allOf:等待所有任务完成
    (public static CompletableFuture allOf(CompletableFuture<?>... cfs))
  2. anyOf:只要有一个任务完成
    (public static CompletableFuture anyOf(CompletableFuture<?>... cfs))

实战演练

案例说明:电商比价需求
同一款产品,同时搜索出同款产品在各大电商的售价;
同一款产品,同时搜索出本产品在某一个电商平台下,各个入驻门店的售价是多少
出来结果希望是同款产品的在不同地方的价格清单列表,返回一个List
in jd price is 88.05
in pdd price is 86.11
in taobao price is 90.43
(String类型)

代码示例

public class test1 {

    public ExecutorService executorService = new ThreadPoolExecutor(40,100,100,TimeUnit.MINUTES,new ArrayBlockingQueue<>(10000));
    public static List<mall> list = Arrays.asList(
            new mall("jd"),
            new mall("tmall"),
            new mall("taobao"),
            new mall("pdd"),
            new mall("elm")
    );

    public static void main(String[] args) {
        showRes3(list,"mysql");
    }
    public static List<String> showRes(List<mall> list,String productName){
        long start = System.currentTimeMillis();
        List<String> collect = list.stream().map(mall -> String.format(productName + " in %s price is %.2f", mall.getName(), mall.getPrice(productName))).collect(Collectors.toList());
        collect.forEach(System.out::println);
        long end = System.currentTimeMillis();
        System.out.println("耗时:"+(end-start)+"秒");
        return collect;
    }
    public static List<String> showRes2(List<mall> list,String productName){
        long start = System.currentTimeMillis();
        List<String> collect = list.parallelStream().map(mall -> CompletableFuture.supplyAsync(() ->
                String.format(productName + " in %s price is %.2f", mall.getName(), mall.getPrice(productName))
        ).join()).collect(Collectors.toList());
        long end = System.currentTimeMillis();
        collect.forEach(System.out::println);
        System.out.println("耗时:"+(end-start)+"秒");
        return collect;
    }

    public static List<String> showRes3(List<mall> list,String productName){
        long start = System.currentTimeMillis();
        List<String> collect = list.stream().map(mall -> CompletableFuture.supplyAsync(() -> String.format(productName + " in %s price is %.2f", mall.getName(), mall.getPrice(productName))
        )).collect(Collectors.toList()).stream().map(s -> s.join()).collect(Collectors.toList());
        long end = System.currentTimeMillis();
        collect.forEach(System.out::println);
        System.out.println("耗时:"+(end-start)+"毫秒");
        return collect;
    }

}


class mall{
    private String name;

    public String getName() {
        return name;
    }

    public mall(String name) {
        this.name = name;
    }

    public double getPrice(String productName){
        try {
            TimeUnit.SECONDS.sleep(1);
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
        return ThreadLocalRandom.current().nextDouble()*2+productName.charAt(0);
    }
}

通过上面代码示例结果得知
通过异步线程方法耗时需要1000毫秒左右
而同步线程方法耗时需要5000毫秒
而且当商城店铺越来越多的时候,异步线程耗时不会增加,同步线程耗时会不断增加,异步的优势就显现出来了。

tips
在使用stream流的时候,如果使用的是集合的stream()方法,再使用异步线程时比如使用join()方法会出现串行,和上面方法三一样得使用两次stream流才能实现异步执行,相当于每一步都得使用stream流。
这个时候可以使用parallelStream并行流就可以解决这个问题
但是使用parallelStream会出现一些问题,所以并行流要谨慎使用!!!

并行流的陷阱

  1. 线程安全
    由于并行流使用多线程,则一切线程安全问题都应该是需要考虑的问题,如:资源竞争、死锁、事务、可见性等等。
  2. 线程消费
    在虚拟机启动时,我们指定了worker线程的数量,整个程序的生命周期都将使用这些工作线程;这必然存在任务生产和消费的问题,如果某个生产者生产了许多重量级的任务(耗时很长),那么其他任务毫无疑问将会没有工作线程可用;更可怕的事情是这些工作线程正在进行IO阻塞。

标签:JUC,System,详解,线程,println,CompletableFuture,public,out
From: https://www.cnblogs.com/gen1us/p/17566679.html

相关文章

  • 转:springboot2.0 集成redis服务详解,以及 (Lettuce & Jedis)
    springboot2.0集成redis服务详解,以及(Lettuce&Jedis)   ......
  • switch写法详解
    我们在开发项目中经常遇到对数据的判断进行相应的逻辑(if..else ,三元运算等),Switch语句用来选择多个需要执行的代码块,一定程度上简化了if....else1.语法switch(表达式){casen:代码块break;casen:代码块break;default:默认代码块}2.代码解释计算一次switch......
  • Matplotlib库中,plt.figure()、plt.imshow()、plt.axis()和plt.show()、gca、savefig、
    在Matplotlib库中,plt.figure()、plt.imshow()、plt.axis()和plt.show()是用于绘制和显示图像的常用方法。下面是对每个方法的含义的解释:plt.figure():plt.figure()用于创建一个新的图形对象(Figure),它是绘图的最顶层容器。可以使用该对象进行图形的设置和操作,例如设置图形的大小......
  • 大语言模型的预训练4:指示学习Instruction Learning详解以及和Prompt Learning,In-cont
    大语言模型的预训练[4]:指示学习InstructionLearning:Entailment-oriented、PLMoriented、human-oriented详解以及和PromptLearning,In-contentLearning区别1.指示学习的定义InstructionLearning让模型对题目/描述式的指令进行学习。针对每个任务,单独生成指示,通过在若干个......
  • 大语言模型的预训练[5]:语境学习、上下文学习In-Context Learning:精调LLM、Prompt设计
    大语言模型的预训练[5]:语境学习、上下文学习In-ContextLearning:精调LLM、Prompt设计和打分函数(ScoringFunction)设计以及ICL底层机制等原理详解1.In-ContextLearning背景与定义背景大规模预训练语言模型(LLM)如GPT-3是在大规模的互联网文本数据上训练,以给定的前缀来预测生......
  • 带你玩转自定义view系列--Android画笔的详解
    View的简介View是Android所有控件的基类,接下来借鉴网上的一张图片让大家一目了然(图片出自:http://blog.51cto.com/wangzhaoli/1292313)imageAndroid画笔的详解Android提供了2D图形绘制的各种工具,如Canvas(画布)、Point(点)、Paint(画笔)、Rectangles(矩形)等,利用这些工具可以直接在......
  • 操作符详解
    写在前面这里的内容虽然有些多,不过整体较为简单,我总结了一些相对有些难度的的知识点。操作符C语言的操作符有很种,这里我把常见的一些和大家进行分析一下.算术操作符移位操作符位操作符赋值操作符算术操作符所谓的算数操作符就是我们的加减乘除,没有什么可以谈的.+-*/......
  • 大语言模型的预训练[2]:GPT、GPT2、GPT3、GPT3.5、GPT4相关理论知识和模型实现、模型
    大语言模型的预训练[2]:GPT、GPT2、GPT3、GPT3.5、GPT4相关理论知识和模型实现、模型应用以及各个版本之间的区别详解1.GPT模型1.1GPT模型简介在自然语言处理问题中,可从互联网上下载大量无标注数据,而针对具体问题的有标注数据却非常少,GPT是一种半监督学习方法,它致力于用大量......
  • 大语言模型的预训练[2]:GPT、GPT2、GPT3、GPT3.5、GPT4相关理论知识和模型实现、模型
    大语言模型的预训练[2]:GPT、GPT2、GPT3、GPT3.5、GPT4相关理论知识和模型实现、模型应用以及各个版本之间的区别详解1.GPT模型1.1GPT模型简介在自然语言处理问题中,可从互联网上下载大量无标注数据,而针对具体问题的有标注数据却非常少,GPT是一种半监督学习方法,它致力于用大......
  • PHP-FPM的配置详解
    1php-fpm的配置详解  和LAMP不同的是,在LNMP架构中,php-fpm作为独立的一个服务存在,既然是独立服务,那么它必然有自己的配置文件。php-fpm的配置文件为/usr/local/php-fpm/etc/php-fpm.conf,它同样也支持include语句,类似于nginx.conf里面的include。  Nginx可以配置多个虚拟......