• 以为很熟悉CountDownLatch的使用了,居然在生产环境翻车了


     

    前言

    万万没想到,在生产环境中竟然翻车了,因为没有考虑到一些场景,导致了CountDownLatch出现了问题,接下来来分享一下由于CountDownLatch导致的问题。

    需求背景

    先简单介绍下业务场景,针对用户批量下载的文件进行修改上传

    为了提高执行的速度,所以在采用线程池去执行 下载-修改-上传 的操作,并在全部执行完之后统一提交保存文件地址到数据库,于是加入了CountDownLatch来进行控制。

    具体实现

    根据服务本身情况,自定义一个线程池

    1. public static ExecutorService testExtcutor() {
    2.        return new ThreadPoolExecutor(
    3.                2,
    4.                2,
    5.                0L,
    6.                TimeUnit.SECONDS,
    7.                new LinkedBlockingQueue<>(1));
    8.   }
    9. 复制代码

    模拟执行

    1. public static void main(String[] args) {
    2.       // 下载文件总数
    3.       List<Integer> resultList = new ArrayList<>(100);
    4.       IntStream.range(0,100).forEach(resultList::add);
    5.       // 下载文件分段
    6.       List<List<Integer>> split = CollUtil.split(resultList, 10);
    7.       ExecutorService executorService = BaseThreadPoolExector.testExtcutor();
    8.       CountDownLatch countDownLatch = new CountDownLatch(100);
    9.       for (List<Integer> list : split) {
    10.           executorService.execute(() -> {
    11.               list.forEach(i ->{
    12.                   try {
    13.                       // 模拟业务操作
    14.                       Thread.sleep(500);
    15.                       System.out.println("任务进入");
    16.                   } catch (InterruptedException e) {
    17.                       e.printStackTrace();
    18.                       System.out.println(e.getMessage());
    19.                   } finally {
    20.                       System.out.println(countDownLatch.getCount());
    21.                       countDownLatch.countDown();
    22.                   }
    23.               });
    24.           });
    25.       }
    26.       try {
    27.           countDownLatch.await();
    28.           System.out.println("countDownLatch.await()");
    29.       } catch (InterruptedException e) {
    30.           e.printStackTrace();
    31.       }
    32.   }
    33. 复制代码

    一开始我个人感觉没有什么问题,反正finally都能够做减一的操作,到最后调用await方法,进行主线程任务

    1. Exception in thread "main" java.util.concurrent.RejectedExecutionException: Task java.util.concurrent.FutureTask@300ffa5d rejected from java.util.concurrent.ThreadPoolExecutor@1f17ae12[Running, pool size = 2, active threads = 2, queued tasks = 1, completed tasks = 0]
    2. at java.util.concurrent.ThreadPoolExecutor$AbortPolicy.rejectedExecution(ThreadPoolExecutor.java:2063)
    3. at java.util.concurrent.ThreadPoolExecutor.reject(ThreadPoolExecutor.java:830)
    4. at java.util.concurrent.ThreadPoolExecutor.execute(ThreadPoolExecutor.java:1379)
    5. at java.util.concurrent.AbstractExecutorService.submit(AbstractExecutorService.java:112)
    6. at Thread.executor.executorTestBlock.main(executorTestBlock.java:28)
    7. 任务进入
    8. countDownLatch.countDown
    9. 任务进入
    10. countDownLatch.countDown
    11. 任务进入
    12. countDownLatch.countDown
    13. 任务进入
    14. countDownLatch.countDown
    15. 任务进入
    16. countDownLatch.countDown
    17. 任务进入
    18. countDownLatch.countDown
    19. 任务进入
    20. countDownLatch.countDown
    21. 任务进入
    22. countDownLatch.countDown
    23. 任务进入
    24. countDownLatch.countDown
    25. 任务进入
    26. countDownLatch.countDown
    27. 任务进入
    28. countDownLatch.countDown
    29. 任务进入
    30. countDownLatch.countDown
    31. 任务进入
    32. countDownLatch.countDown
    33. 任务进入
    34. countDownLatch.countDown
    35. 任务进入
    36. countDownLatch.countDown
    37. 复制代码

    由于任务数量较多,阻塞队列中已经塞满了,所以默认的拒绝策略,当队列满时,处理策略报错异常,

    要注意这个异常是线程池,自己抛出的,不是我们循环里面打印出来的,

    这也造成了,线上这个线程池被阻塞了,他永远也调用不到await方法,

    利用jstack,我们就能够看到有问题

    解决方案

    1. 调大阻塞队列,但是问题来了,到底多少阻塞队列才是大呢,如果太大了会不由又造成内存溢出等其他的问题

    2. 在第一个的基础上,我们修改了拒绝策略,当触发拒绝策略的时候,用调用者所在的线程来执行任务

      1. public static ThreadPoolExecutor queueExecutor(BlockingQueue workQueue){
      2.       return new ThreadPoolExecutor(
      3.               size,
      4.               size,
      5.               0L,
      6.               TimeUnit.SECONDS,
      7.               workQueue,
      8.               new ThreadPoolExecutor.CallerRunsPolicy());
      9.   }
      10. 复制代码
    3. 你可能又会想说,会不会任务数量太多,导致调用者所在的线程执行不过来,任务提交的性能急剧下降

      那我们就应该自定义拒绝策略,将这下排队的消息记录下来,采用补偿机制的方式去执行

    4. 同时也要注意上面的那个异常是线程池抛出来的,我们自己也需要将线程池进行try catch,记录问题数据,并且在finally中执行countDownLatch.countDown来避免,线程池的使用

    总结

    目前根据业务部门的反馈,业务实际中任务数不很特别多的情况,所以暂时先采用了第二种方式去解决这个线上问题

    在这里我们也可以看到,如果没有正确的关闭countDownLatch,可能会导致一直等待,这也是我们需要注意的。

    工具虽然好,但是依然要注意他带来的问题,没有正确的去处理好,引发的一系列连锁反应。

  • 相关阅读:
    Python树莓派开发
    【JavaEE初阶】 JavaScript基础语法——贰
    实习项目遇到的bug
    因为axios请求后端,接收不到token的问引出的问题
    k8s~envoy上添加wasm插件
    MyBatis 及 MyBatis Plus 纯注解方式配置(Spring Boot + Postgresql)
    王道操作系统___第四章02
    QT MV\MVC结构
    Transformer - Attention Is All You Need - 跟李沐学AI
    docker安装mysql数据库,忽略大小写,设置时区
  • 原文地址:https://blog.csdn.net/YYniannian/article/details/126221115