长连接中异步任务的同步等待陷阱:一次主线程阻塞的排查与修复

最近的项目出现了一个bug,长连接中使用了异步线程池,具体是有一个循环,每一次循环都用了异步线程,然后呢,在长连接中处理这个循环的同时,还会有前端向后端发送消息,那么后端也要接收相应的消息进行处理,每次循环的异步线程如下

java 复制代码
Future<Boolean> submit = taskExecutor.submit(new Callable<Boolean>() {
                @Override
                public Boolean call(){}
                });

然后呢还搞了一个

java 复制代码
private final Map<String, Future<Boolean>> Futures = new ConcurrentHashMap<>();
Futures.put(id, submit);

以id为key保存每一次循环中的异步线程返回值,然后呢,最灵魂的代码来喽,这句代码真是给添了不少麻烦,那就是等待这次异步任务完成,实际上大错特错

java 复制代码
Future<Boolean> future = Futures.get(id);
Boolean b = future.get();

这句代码乍一看倒是没什么问题,但是呢,在运行的时候,前端向后端发送特定的消息,后端没有响应,哇靠,这是为啥嘞?

没错,就是Boolean b = future.get();把主线程给阻塞了,导致不能处理其他问题,其实这只是一个小问题哈,但是有时候往往是小问题在写代码的时候,加着加着就忘记了,结果排查的时候好半天才看到。

最后的解决办法是啥呢,最后搞了一个异步线程处理这个循环,大致的代码示意如下:

java 复制代码
 CompletableFuture.runAsync(()->
    {
        for (dto:list) {
            Future<Boolean> submit = taskExecutor.submit(new Callable<Boolean>() {
                @Override
                public Boolean call() {
                }
            });
        }
    })
相关推荐
二哈赛车手1 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
栗子~~1 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
YDS8292 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
未若君雅裁3 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis
AI人工智能+电脑小能手3 小时前
【大白话说Java面试题 第65题】【JVM篇】第25题:谈谈对 OOM 的认识
java·开发语言·jvm
阿维的博客日记4 小时前
Nacos 为什么能让配置动态生效?(涉及 @RefreshScope 注解)
java·spring
雨辰AI4 小时前
SpringBoot3 + 人大金仓读写分离 + 分库分表 + 集群高可用 全栈实战
java·数据库·mysql·政务
辰海Coding5 小时前
MiniSpring框架学习-完成的 IoC 容器
java·spring boot·学习·架构
小小编程路5 小时前
C++ 多线程与并发
java·jvm·c++
AI视觉网奇5 小时前
linux 检索库 判断库是否支持
java·linux·服务器