Java线程池异步CompletableFuture,创建一步对象、回调方法、线程串行化、两任务组合、多任务组合

        JDK1.8后引入的 CompletableFuture 类,提供了一种更为便捷的异步编程方式,可以轻松地进行异步操作的组合和串联。

        CompletableFuture实现了Future类,可以获取到异步结果。

创建异步对象

conpletableFuture提供了四个静态方法来创建一个异步的操作。

static CompletableFuture<Void> runAsync(Runnable runnable)

static CompletableFuture<Void> runAsync(Runnable runnable, Executor executor)

static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier)

static <U> CompletableFuture<U> supplyAsync(Supplier<U> supplier, Executor executor)

1.runXxx都没有返回结果,supplyXxx都是可以获取返回结果

2.可以传入自定义线程池,否则默认线程池。 

默认线程池为ForkJoinPool

简单事例

// 创建线程池
    static ExecutorService executorService = Executors.newFixedThreadPool(10);


    public static void main(String[] args) throws ExecutionException, InterruptedException {
        CompletableFuture<Integer> integerCompletableFuture = CompletableFuture.supplyAsync(() -> {
            System.out.printf("当前线程:" + Thread.currentThread().getId());
            int i = 10 / 2;
            System.out.printf("运行结果" + i);  
            return i;
        }, executorService);
        System.out.printf("\n启动结束----"+integerCompletableFuture.get());
    }

 回调方法

方法完成后的感知

CompletableFuture<T> whenComplete(BiConsumer<? super T, ? super Throwable> action)
CompletableFuture<T> whenCompleteAsync(BiConsumer<? super T, ? super Throwable> action)
CompletableFuture<T> whenCompleteAsync(BiConsumer<? super T, ? super Throwable> action, Executor executor)
CompletableFuture<T> exceptionally(Function<Throwable, ? extends T> fn)

带Async意思,前面执行成功后,我们还要干一件事情,交给线程池处理。

不带Async ,前后使用同一个线程处理。

方法不以Async结尾的,意味着Action使用相同的线程执行,而Async可能会使用其他线程执行(如果使用相同的线程池,也可能会被同一个线程执行) 

whenComplete可以处理正常和异常的计算结果,exceptionally处理异常情况

简单事例

// 创建线程池
    static ExecutorService executorService = Executors.newFixedThreadPool(10);


    public static void main(String[] args) throws ExecutionException, InterruptedException {
        CompletableFuture<Integer> integerCompletableFuture = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n当前线程:" + Thread.currentThread().getId());
            int i = 10 / 0;
            System.out.printf("\n运行结果" + i);
            return i;
        }, executorService).whenCompleteAsync((t,u)->{
            System.out.printf("\n异步任务完成,结果是:"+t+"异常是"+u);
            //whenCompleteAsync 可以得到异常信息,无法修改返回数据
        }).exceptionally(throwable -> {
            // 可以感知异常,同时返回默认值
            return 10;
        });
        System.out.printf("\n启动结束----"+integerCompletableFuture.get());
    }

handle方法

方法执行完成后的处理

与Complete一样,可对结果做最后处理(可处理异常),可改变返回值

<U> CompletableFuture<U> handle(BiFunction<? super T, Throwable, ? extends U> fn)
<U> CompletableFuture<U> handleAsync(BiFunction<? super T, Throwable, ? extends U> fn)
<U> CompletableFuture<U> handleAsync(BiFunction<? super T, Throwable, ? extends U> fn, Executor executor)

简单事例

// 创建线程池
    static ExecutorService executorService = Executors.newFixedThreadPool(10);


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

        CompletableFuture<Integer> integerCompletableFuture = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n当前线程:" + Thread.currentThread().getId());
            int i = 10 / 0;
            System.out.printf("\n运行结果" + i);
            return i;
        }, executorService).handle((res,thr)->{
            if (res !=null){
                return res+1;
            }
            if(thr!=null){
                return 0;
            }
            return res;
        });
        System.out.printf("\n启动结束----"+integerCompletableFuture.get());
    }

线程串行化方法

//不接收上一个执行的结果

CompletableFuture<Void> thenRun(Runnable action )

CompletableFuture<Void> thenRunAsync(Runnable action)

CompletableFuture<Void> thenRunAsync(Runnable action, Executor executor)

// 接收上一个执行的结果

CompletableFuture<Void> thenAccept(Consumer<? super T> action)

CompletableFuture<Void> thenAcceptAsync(Consumer<? super T> action)

CompletableFuture<Void> thenAcceptAsync(Consumer<? super T> action, Executor executor)

//接受上一个执行结果,处理并把处理结果返回 

<U> CompletableFuture<U> thenApply(Function<? super T,? extends U> fn)

<U> CompletableFuture<U> thenApplyAsync(Function<? super T,? extends U> fn)

<U> CompletableFuture<U> thenApplyAsync(Function<? super T,? extends U> fn, Executor executor)

thenRun方法:只要上面任务执行完成,就开始执行thenRun(无法感知到上一个执行结果)。

thenAccept方法:消费处理结果(接收到上一个任务的处理结果),接受任务的处理结果,并消费处理,无返回结果。

thenApply方法:当一个线程依赖另一个线程时,获取上一个任务返回的结果,并返回当前任务的返回值。

简单事例

// 创建线程池
    static ExecutorService executorService = Executors.newFixedThreadPool(10);


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

        // thenRun:不能获取上一步的执行结果
        CompletableFuture<Void> thenRunFuture = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n当前线程:" + Thread.currentThread().getId());
            int i = 10 / 2;
            System.out.printf("\n运行结果" + i);
            return i;
        }, executorService).thenRunAsync(() -> {
            System.out.printf("\n2启动开始----");

        });
        // thenAccept能接收上一步接口,没有返回值
        CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n当前线程:" + Thread.currentThread().getId());
            int i = 10 / 2;
            System.out.printf("\n运行结果" + i);
            return i;
        }, executorService).thenAcceptAsync(res->{
            System.out.printf("\n2启动开始----"+res);
        });

        //thenApply能接收上一步接口,有返回值
        CompletableFuture<Integer> thenApplyFuture = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n当前线程:" + Thread.currentThread().getId());
            int i = 10 / 2;
            System.out.printf("\n运行结果" + i);
            return i;
        }, executorService).thenApplyAsync((t) -> {
            System.out.printf("\n2启动开始----" + t);
            return t + 2;
        });
        System.out.printf("\n结束:"+thenApplyFuture.get());
    }

两任务组合-都要完成

CompletableFuture<Void> runAfterBoth(CompletionStage<?> other, Runnable action)

CompletableFuture<Void> runAfterBothAsync(CompletionStage<?> other, Runnable action)

CompletableFuture<Void> runAfterBothAsync(CompletionStage<?> other, Runnable action, Executor executor)

<U> CompletableFuture<Void> thenAcceptBoth(CompletionStage<? extends U> other, BiConsumer<? super T, ? super U> action)

<U> CompletableFuture<Void> thenAcceptBothAsync(CompletionStage<? extends U> other, BiConsumer<? super T, ? super U> action)

<U> CompletableFuture<Void> thenAcceptBothAsync(CompletionStage<? extends U> other, BiConsumer<? super T, ? super U> action, Executor executor)

<U,V> CompletableFuture<V> thenCombine(CompletionStage<? extends U> other, BiFunction<? super T,? super U,? extends V> fn)

<U,V> CompletableFuture<V> thenCombineAsync(CompletionStage<? extends U> other, BiFunction<? super T,? super U,? extends V> fn)

<U,V> CompletableFuture<V> thenCombineAsync(CompletionStage<? extends U> other, BiFunction<? super T,? super U,? extends V> fn, Executor executor)

runAfterBoth:组合两个future,不需要获取future的结果,只需两个future处理完任务后,处理该任务。

thenAcceptBoth:组合两个future,获取两个future任务的返回结果,然后处理任务,没有返回值。

thenCombine:组合两个future,获取两个future任务的返回结果,然后处理任务,返回当前任务的返回值。

简单事例

// 创建线程池
    static ExecutorService executorService = Executors.newFixedThreadPool(10);


    public static void main(String[] args) throws ExecutionException, InterruptedException {
// 任务1
        CompletableFuture<Integer> future1 = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n任务1线程:" + Thread.currentThread().getId());
            int i = 10 / 2;
            System.out.printf("\n任务1结果" + i);
            return i;
        }, executorService);
//任务2
        CompletableFuture<String> future2 = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n任务2线程:" + Thread.currentThread().getId());
            System.out.printf("\n任务2结果");
            return "Hello";
        }, executorService);
        // runAfterBothAsync 不能感知前两步结果
        future1.runAfterBothAsync(future2,()->{
            System.out.printf("\n任务3");
        },executorService);
        // thenAcceptBothAsync 能拿到前两步的结果
        future1.thenAcceptBothAsync(future2,(f1,f2)->{
            System.out.printf("\n任务3f1:"+f1);
            System.out.printf("\n任务3f2:"+f2);

        },executorService);
        // stringCompletableFuture 能拿到前两步的结果并返回结果
        CompletableFuture<String> stringCompletableFuture = future1.thenCombineAsync(future2, (f1, f2) -> {
            System.out.printf("\n任务3f1:" + f1);
            System.out.printf("\n任务3f2:" + f2);
            return f1 + f2;

        }, executorService);
        System.out.printf("----"+stringCompletableFuture.get());
    }

两任务组合-一个完成

两个任务中,任意一个future完成,执行任务。 

CompletableFuture<Void> runAfterEither(CompletionStage<?> other, Runnable action)

CompletableFuture<Void> runAfterEitherAsync(CompletionStage<?> other, Runnable action)

CompletableFuture<Void> runAfterEitherAsync(CompletionStage<?> other, Runnable action,  Executor executor)

CompletableFuture<Void> acceptEither( CompletionStage<? extends T> other, Consumer<? super T> action)

CompletableFuture<Void> acceptEitherAsync(CompletionStage<? extends T> other, Consumer<? super T> action)

CompletableFuture<Void> acceptEitherAsync(CompletionStage<? extends T> other, Consumer<? super T> action, Executor executor)

<U> CompletableFuture<U> applyToEither(CompletionStage<? extends T> other, Function<? super T, U> fn)

<U> CompletableFuture<U> applyToEitherAsync(CompletionStage<? extends T> other, Function<? super T, U> fn)

<U> CompletableFuture<U> applyToEitherAsync(CompletionStage<? extends T> other, Function<? super T, U> fn, Executor executor)

 runAfterEither:两个任务有一个执行完成,不获取返回值。

 acceptEither:两个任务有一个执行完成,获取返回值,没有新的返回值。

 applyToEither:两个任务有一个执行完成,获取返回值,处理任务并有新的返回值。

简单事例

 // 创建线程池
    static ExecutorService executorService = Executors.newFixedThreadPool(10);


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

        CompletableFuture<Object> future1 = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n任务1线程:" + Thread.currentThread().getId());
            int i = 10 / 2;
            System.out.printf("\n任务1结果" + i);
            return i;
        }, executorService);
        CompletableFuture<Object> future2 = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n任务2线程:" + Thread.currentThread().getId());
            try {
                Thread.sleep(3000);
                System.out.printf("\n任务2结果");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return "Hello";
        }, executorService);
        // runAfterEitherAsync 不感知结果,没有返回值。
        future1.runAfterEitherAsync(future2,()->{
            System.out.printf("\n任务3");
        },executorService);
        // acceptEitherAsync 感知结果,没有返回值。
        future1.acceptEitherAsync(future2,(res)->{
            System.out.printf("\n任务3:"+res);
        },executorService);
        // applyToEitherAsync 感知结果,有返回值。
        CompletableFuture<String> stringCompletableFuture = future1.applyToEitherAsync(future2, (res) -> {
            System.out.printf("\n任务3:" + res);
            return res.toString() + "aaaa";
        }, executorService);
        System.out.printf("最终结果"+stringCompletableFuture.get());
    }

多任务组合

static CompletableFuture<Void> allOf(CompletableFuture<?>... cfs)

static CompletableFuture<Object> anyOf(CompletableFuture<?>... cfs) 

allOf:等待所有任务完成。

anyOf:只要有一个任务完成。

简单事例

// 创建线程池
    static ExecutorService executorService = Executors.newFixedThreadPool(10);


    public static void main(String[] args) throws ExecutionException, InterruptedException {
        CompletableFuture<String> future1 = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n查询商品名称");
            return "手机";

        },executorService);
        CompletableFuture<String> future2 = CompletableFuture.supplyAsync(() -> {
            System.out.printf("\n查询商品颜色");
            return "黑色";
        },executorService);
        CompletableFuture<String> future3 = CompletableFuture.supplyAsync(() -> {
            try {
                Thread.sleep(3000);
                System.out.printf("\n查询商品品牌");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return "华为";

        },executorService);
        CompletableFuture<Void> future = CompletableFuture.allOf(future1, future2, future3);
        // 等待所有结果全部完成,如果不加future.get();,顺序是乱的
        future.get();
        System.out.printf("\nallOf完成");
        System.out.printf("\nallOf完成"+future3.get()+future1.get()+future2.get());

        CompletableFuture<Object> anyOf = CompletableFuture.anyOf(future1, future2, future3);
        System.out.printf("\nanyOf完成");
        System.out.printf("\nanyOf完成"+anyOf.get());
    }

如果对您有帮助辛苦帮忙点赞收藏!!! 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值