美文网首页
CompletableFuture避坑2——allOf()超时时

CompletableFuture避坑2——allOf()超时时

作者: 猫尾草 | 来源:发表于2021-08-01 21:24 被阅读0次

CompletableFuture避坑1——需要自定义线程池
CompletableFuture避坑2——allOf()超时时间不合理的后果
CompletableFuture避坑3——线程池的DiscardPolicy()导致整个程序卡死


运行下面这段程序

import com.alibaba.fastjson.JSONObject;
import com.google.common.util.concurrent.ThreadFactoryBuilder;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

public class ThreadPoolTest {

    private static final Logger log = LoggerFactory.getLogger(ThreadPoolTest.class);

    public static void main(String[] args) {
        List<CompletableFuture> futures = new ArrayList<>(3);
        CompletableFuture first = CompletableFuture.runAsync(() -> {
            try {
                Thread.sleep(5000);
                System.out.println("first");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }).exceptionally(e -> {
            e.printStackTrace();
            return null;
        });
        futures.add(first);
        Long startTime = System.currentTimeMillis();
        log.info("开始等待所有线程结束,时间:{}", startTime);
        try {
            CompletableFuture.allOf(futures.toArray(new CompletableFuture[0])).orTimeout(2000, TimeUnit.MILLISECONDS).join();
        } catch (Exception e) {
            log.error("等待所有线程结束发生异常:", e);
        } finally {
            Long endTime = System.currentTimeMillis();
            log.info("等待结束,时间:{}, 耗时:{}秒", startTime, (endTime-startTime)/1000);
            try {
                log.info(JSONObject.toJSONString(first));
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
}

输出

10:57:59.612 [main] INFO com.example.demo.ThreadPoolTest2 - 开始等待所有线程结束,时间:1627268279609
10:58:01.642 [main] ERROR com.example.demo.ThreadPoolTest2 - 等待所有线程结束发生异常:
java.util.concurrent.CompletionException: java.util.concurrent.TimeoutException
    at java.base/java.util.concurrent.CompletableFuture.reportJoin(CompletableFuture.java:412)
    at java.base/java.util.concurrent.CompletableFuture.join(CompletableFuture.java:2044)
    at com.example.demo.ThreadPoolTest2.main(ThreadPoolTest2.java:46)
Caused by: java.util.concurrent.TimeoutException: null
    at java.base/java.util.concurrent.CompletableFuture$Timeout.run(CompletableFuture.java:2792)
    at java.base/java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:515)
    at java.base/java.util.concurrent.FutureTask.run(FutureTask.java:264)
    at java.base/java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.run(ScheduledThreadPoolExecutor.java:304)
    at java.base/java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1128)
    at java.base/java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:628)
    at java.base/java.lang.Thread.run(Thread.java:834)
10:58:01.643 [main] INFO com.example.demo.ThreadPoolTest2 - 等待结束,时间:1627268279609, 耗时:2秒
10:58:01.726 [main] INFO com.example.demo.ThreadPoolTest2 - {"cancelled":false,"completedExceptionally":false,"done":false,"numberOfDependents":2}
10:58:04.309 [HomePageCardFeatureQueryThreadPool-0] INFO com.example.demo.ThreadPoolTest2 - 线程first结束

可以看到,设置allof的等待时间(2s)比线程实际执行时间(5s)短,等待超时结束时,线程并没有被杀死,也没有被取消,也没有抛异常,这个线程继续运行着,直到正常结束。
这会有什么问题呢?
如果接口中存在这种代码,大量请求超时返回,但是实际子线程还在运行,接口又接收更多的请求,继续创建子线程(没有使用线程池)或者进入线程池排队,前者会导致 OOM,后者则会加剧接口超时(任务都在排队,等不到执行)导致接口乃至整个服务完全不可用。
实际代码中我们有很多io操作,一般都会设置超时时间(不设置的后果很严重),如果子线程中调用了这些方法,那么allof()方法的超时时间一定要大于最大的超时时间(如果有串行操作,还需要累加)。

相关文章

  • CompletableFuture 组合处理 allOf 和 a

    allOf 和 anyOf 可以组合任意多个 CompletableFuture。函数接口定义如下所示。首先,这两...

  • java8 CompletableFuture,allOf多实例

    我们在处理业务时,有时会有多任务异步处理,同步返回结果的情况,在java中,我们可以使用CompletableFu...

  • CompletableFuture.allOf 可变参数传参

  • completablefuture并发任务

    利用allof方法

  • 面试回答

    JAVA 1. completableFuture 2. stream

  • 避坑

    Day10/21 【21天阅读挑战】 【书籍名称】 《我不是教你诈1》 作者:[美]刘墉 【阅读感受】 又到了毕业...

  • 避坑

    人这一辈子着实不容易,有时太平顺的路走着走着,忘了低头看路,就掉进坑里。有些坑浅,磕碰得灰头土脸鼻青脸肿的,...

  • 避坑

    前阵子读一本书 说中国的男人很惨 没有更年期 我就问我爸爸男人有没有更年期 他说没有 然后想起以前不会业务的老大 ...

  • [避坑]

    1 组件中引用canvas【一点要加this】 2 如在组件外调用组件内的canvas需要绑定上下文

  • 避坑

    1、 账号昵称、头像、个性签名、主页背景图等都不能出现违规内容。2、 新号一开始不要发布太长视频,完播率会很差,难...

网友评论

      本文标题:CompletableFuture避坑2——allOf()超时时

      本文链接:https://www.haomeiwen.com/subject/bktnmltx.html