CompletableFuture是java8引入的新类,该类实现了 Future 接口和 CompletionStage 接口,封装了future、forkjoin相关类来执行异步,下面是使用demo
@Test
public void testPool() throws ExecutionException, InterruptedException {
System.out.println(Runtime.getRuntime().availableProcessors());
ThreadPoolExecutor executor = new ThreadPoolExecutor(5, 100, 2, TimeUnit.MINUTES, new LinkedBlockingQueue<>(100),
Thread::new, new ThreadPoolExecutor.CallerRunsPolicy());
executor.allowCoreThreadTimeOut(false);
List<CompletableFuture<String>> completableFutures = new ArrayList<>();
for (int i =1; i<=10;i++) {
int finalI = i;
CompletableFuture<String> voidCompletableFuture = CompletableFuture.supplyAsync(() -> {
try {
System.out.println("线程:"+ finalI + "执行开始:" + System.currentTimeMillis());
Thread.sleep(5000);
System.out.println("线程:"+ finalI + "执行结束" + System.currentTimeMillis());
} catch (InterruptedException e) {
e.printStackTrace();
}
return ""+finalI;
}, executor);
completableFutures.add(voidCompletableFuture);
}
completableFutures.forEach(CompletableFuture::join);
// 获取随便获取返回结果
System.out.println(completableFutures.get(1).get());
}
创建的线程池的核心数,由服务器的cup核数来参考
看线程内执行的内容来分别
如服务器的cup核数为N
- IO密集型,主要操作磁盘或者远程调用接口 ,配置为2N
- CPU密集型,主要进行一些循环,递归计算等功能,配置为N+1
上面代码例子打印出来的结果就是
踩坑记录:
第一次使用时,将completableFutures.forEach(CompletableFuture::join);放在循环里面,导致多线程未起作用,而是循环内执行完成,才到下一个去执行。