java 8 CompletableFuture (1)

组合式异步编程实践
本文介绍了使用Java实现组合式异步编程的方法,包括通过Future和CompletableFuture进行远程数据获取及处理的技术细节,并对比了不同实现方式的性能差异。

组合式异步编程

 

package com.example.demo.future;

import org.junit.jupiter.api.Test;

import java.util.List;
import java.util.UUID;
import java.util.concurrent.*;
import java.util.stream.Collectors;
import java.util.stream.IntStream;

public class FutureTest {

    /**
     * 运程数据
     *
     * @return
     */
    private static String data() {
        try {

            Thread.sleep(1020); //拟处理远程处理时间
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return String.format("{id:%s,orderNo:'%s'}", 
               Math.round(Math.random() * 10000), UUID.randomUUID().toString());
    }

    /**
     * 获取远程的订单
     *
     * @return
     */
    Future<String> getRemoteOrder() {
        ExecutorService executorService = Executors.newCachedThreadPool();
        return executorService.submit(FutureTest::data);
    }

    /**
     * 第二种实现
     *
     * @return
     */
    Future<String> getRemoteOrderII() {
        CompletableFuture<String> future = new CompletableFuture<>();
        new Thread(() -> {
            try {
                future.complete(data());  //正常获取值
            } catch (Exception e) {
                future.completeExceptionally(e);    //以异常抛出结束
            }
        }).start();
        return future;
    }

    /**
     * 第三种实现
     * 这种方法完全等价于第二种 ,低层会交由 ForkJoinPool 去执行。你也可以使用自己的 Executor
     *
     * @return
     */
    Future<String> getRemoteOrderIII() {
        return CompletableFuture.supplyAsync(FutureTest::data);
    }

    @Test
    public void test1() throws InterruptedException, ExecutionException, TimeoutException {

        // 传统方法
        Future<String> userDataFuture = getRemoteOrder();
        userDataFuture.isDone();  //是否请求完
        //   orderFuture.get();   //禁止这样写,防止线程非正常结束后,这里将会无限阻塞
        String i = userDataFuture.get(10, TimeUnit.SECONDS); //最多阻塞10秒,就算未完成,也返回结果


        //CompletableFuture 方法
        Future<String> orderFuture = getRemoteOrderII();
        orderFuture.get(5, TimeUnit.SECONDS);  //最多阻塞10秒,就算未完成,也返回结果


    }

    /**
     * 并发测试1
     */
    @Test
    public void test2() {
        //顺序进行
        long beginTime = System.currentTimeMillis();

        List<String> orders = IntStream.range(1, 5)
                .mapToObj(i -> data())
                .collect(Collectors.toList());

        long endTime = System.currentTimeMillis();

        System.out.println("串行发费时间 :" + (endTime - beginTime) + " ms"); //4090 ms

        //并行
        beginTime = System.currentTimeMillis();

        orders = IntStream.range(1, 5)
                .parallel()
                .mapToObj(i -> data())
                .collect(Collectors.toList());

        endTime = System.currentTimeMillis();

        System.out.println("并行发费时间 :" + (endTime - beginTime) + " ms"); //1027 ms


        //并行,第二种方式
        beginTime = System.currentTimeMillis();

        List<CompletableFuture<String>> futures = IntStream.range(1, 5)
                .mapToObj(i -> CompletableFuture.supplyAsync(FutureTest::data))
                .collect(Collectors.toList());

        orders = futures.stream().map(CompletableFuture::join) 
 //CompletableFuture.join 与 CompletableFuture.get 一样,只是join不会抛出任何检测到的异常
                .collect(Collectors.toList());

        endTime = System.currentTimeMillis();


        System.out.println("并行II发费时间 :" + (endTime - beginTime) + " ms"); //1023 ms
    }

   

}

 

1.并行第二种式第一种都是使用ForkJoin,所以速度都差不多。但第二种可以设置你自己的线程执行器,改变线程池的数量
  2.线程池大小与处理器的利用率之比可以使用下面的公式进行估算:
  N(threads) = N(cpu) * U(cpu) * (1+W/C)
   N(cpu): 是处理器的核的数目,可以 通过Runtime.getRuntime().availableProcessors() 得到
   U(cpu):  是期望的cpu利用率(该值应该介于 0 - 1之间)
   W/C :    是等待时间 与计算时间的比率

转载于:https://my.oschina.net/u/2552286/blog/1864101

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值