查看原文
其他

记一次线程池使用不当导致触发死锁导致RocketMQ消费停滞

weihubeats 小奏技术
2024-09-06

大家好,这里是小奏,觉得文章不错可以关注公众号小奏技术

背景

团队小伙伴突然找到我们团队说,不得了了,线上的RocketMQ又出现了消费停滞,怎么办? 要不要我们先重启一下

其实早在之前也出现过一次,当时为了快速恢复业务的,就直接重启解决的。

这次因为堆积量不多,所以想对运行环境进行一些环境快照保留。所以就和业务方沟通是否看见接受短暂的消息延时

得到肯定答案后就是放手干吧

问题定位

首先我们要确定业务反馈的是否属实,所以需要去RocketMQ dashboard上看看具体的消费进度

可以看到consumer中并不是全部的queue消费都堆积了,只有一个queuq消息堆积了。

这个消费者订阅的topic是分区有序的,正常来说分区有序,如果某个分区的消息单条消息出现了消费异常,必须要等这条消息消费成功(或者是重试结束)后才能继续消费后面的消息。

有时候会因为这个原因出现消息堆积是正常的,但是业务对消息重试进行了合理的设置,设置的重试次数比较合理,不会出现长时间的堆积

RocketMQ的消费线程

一般出现这种问题很明显就是线程出现了死锁或者僵死之类的情况

熟悉RocketMQ的都知道RocketMQ消费消息主要是依赖1个线程1个线程池

  1. PullMessageService开头的线程, 主要用来拉去消息
  1. ConsumeMessageThread开头的线程(实际是一个线程池),主要用来执行消费逻辑

直到了RocketMQ的消费线程模型后我们就好解决了。我们直接通过jstack命令查看线程的堆栈信息

线程快照分析

我们直接通过jstack命令生成线程快照

jstack <pid> > thread_dump_$(date +%Y%m%d_%H%M%S).txt

pid 和后面的 thread_dump_$(date +%Y%m%d_%H%M%S).txt自己随便取个名字就行。自己记得就行

由于应用运行在pod中,生成了我们就下载到本地

我们自己看还是比较难分析出分体。这里我们直接使用一个在线的网站进行线程快照的分析

fastthread

fastthread是一个在线的线程快照分析工具,可以直接将线程快照上传到这个网站进行分析

我们上传我们下载的线程快照文件

然后进行线程分析

很快定位到阻塞其他线程的代码

这里的代码被我打码了

arhtas

如果我们使用arthas也可以很方便的找到阻塞的线程

thread -b

arthas 提供了thread -b, 一键找出那个罪魁祸首。

问题元凶找到

通过阻塞代码我们很快定位到是由于线程池使用不当导致的阻塞

线程池使用不当

什么情况下会出现线程池使用不当导致的"死锁"呢?

我们看看下面的demo

public class XiaoZouExample {

    public static void main(String[] args) {

        ExecutorService executor = new ThreadPoolExecutor(250L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<>());


        // 提交两个外部任务
        for (int i = 0; i < 2; i++) {
            executor.submit(new OuterTask(executor));
        }

        // 等待一段时间后关闭线程池
        try {
            Thread.sleep(10000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        executor.shutdown();

    }

    static class OuterTask implements Runnable {

        private final ExecutorService executor;

        public OuterTask(ExecutorService executor) {
            this.executor = executor;
        }

        @Override
        public void run() {
            System.out.println("小奏技术 Outer task started by thread: " + Thread.currentThread().getName());

            // 创建一个Future来等待内部任务的结果
            Future<?> future = executor.submit(new InnerTask());

            try {
                // 等待内部任务完成
                future.get();
            } catch (InterruptedException | ExecutionException e) {
                e.printStackTrace();
            }

            System.out.println("小奏技术 Outer task finished by thread: " + Thread.currentThread().getName());
        }
    }

    static class InnerTask implements Runnable {
        @Override
        public void run() {
            System.out.println("小奏技术 Inner task started by thread: " + Thread.currentThread().getName());
            try {
                // 模拟长时间运行的任务
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("小奏技术 Inner task finished by thread: " + Thread.currentThread().getName());
        }
    }

}
  • 运行结果

可以看到没有任何任务执行完成,线程池一直处于被阻塞状态

核心原因就是首先线程池的核心线程数是2,核心线程用来执行2个任务,用完了所有线程

然后在核心线程执行的2个任务中又用原来的线程池进行执行任务,这时候因为没有线程可以去执行任务了,所以会添加到阻塞队列中等待核心线程执行完任务后再执行

但是核心线程想要释放任务又必须等待这两个子任务执行完,这样就形成了一个死锁

解决方案

解决方式有多种,最简单的方式可以考虑不要使用队列,直接使用SynchronousQueue

ExecutorService executor = new ThreadPoolExecutor(250L, TimeUnit.MILLISECONDS, new SynchronousQueue<>());

让多的任务直接通过主线程执行或者丢弃任务

当然最好的方式就是避免这种情况的发生,合理的使用线程池,不要线程池中的任务还要使用这个线程池去执行任务

这种情况是比较难避免的,因为现在大部分业务开发都是隐式使用线程池,自己也不知道自己用的哪个线程池

比如spring的@Async注解,@Scheduled注解等等

总结

线程池的使用还是要尽量消息,要避免出现线程池中的任务继续使用该线程池去执行任务,出现死锁

也可以考虑对线程池进行监控,避免出现大量任务阻塞

这个问题想要复现需要大量任务并且超过核心线程数才能复现,还是比较难复现的,只有线上大流量的时候才能复现


继续滑动看下一个
小奏技术
向上滑动看下一个

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存