0

0

Java并发编程中如何编写高可用代码_实践经验总结

P粉602998670

P粉602998670

发布时间:2026-01-22 10:09:20

|

673人浏览过

|

来源于php中文网

原创

高并发下synchronized易致服务不可用,因其在长耗时i/o时阻塞线程引发排队雪崩;应减小锁粒度、移出临界区的阻塞操作,并优选reentrantlock或cas机制。

java并发编程中如何编写高可用代码_实践经验总结

为什么直接用 synchronized 容易导致服务不可用

不是锁本身有问题,而是它在高并发、长耗时场景下会把线程卡死在临界区外,形成“排队雪崩”。比如一个接口里用 synchronized 包裹了远程 HTTP 调用,一旦下游响应变慢或超时,所有后续请求全堵在锁入口,TPS 断崖下跌。

  • 锁粒度要细:优先锁对象字段,而不是整个方法或 this
  • 避免在同步块内做 I/O、RPC、数据库查询——这些操作应移出临界区
  • 考虑用 ReentrantLock 替代,它支持超时获取(tryLock(long, TimeUnit)),失败可快速降级或重试
  • 若必须同步远程调用结果,改用缓存 + CAS 更新(如 AtomicReference.compareAndSet)更安全

ConcurrentHashMap 的常见误用与扩容陷阱

很多人以为 ConcurrentHashMap 是“万能线程安全 Map”,但它的线程安全仅限于单个操作原子性,组合操作(如“检查再插入”)仍需额外同步。

  • computeIfAbsent 是安全的,但注意其 lambda 内不能有阻塞逻辑,否则会拖慢整个 segment 的写入
  • 扩容期间读操作不阻塞,但写操作可能触发帮助扩容(helpTransfer),CPU 使用率会阶段性飙升
  • 初始化容量别设太小:new ConcurrentHashMap(16) 在写入 1000 条后大概率触发多次扩容;建议预估 size 后乘以 1.5 再向上取最近 2 的幂
  • 不要用 keySet().iterator() 做遍历并修改,迭代器弱一致性不保证看到最新写入,且无法检测并发修改异常

线程池配置不当引发的 OOM 和拒绝风暴

线上最常踩的坑是把 Executors.newFixedThreadPool(n) 直接丢进生产环境。它的 LinkedBlockingQueue 默认无界,任务堆积时内存持续上涨,直到 GC 失败或被 OS Kill。

Kacha
Kacha

KaCha是一款革命性的AI写真工具,用AI技术将照片变成杰作!

下载
  • 永远手动构造 ThreadPoolExecutor,明确指定有界队列(如 ArrayBlockingQueue)和拒绝策略
  • 拒绝策略别只用 AbortPolicy(抛 RejectedExecutionException),对核心链路建议用 CallerRunsPolicy 让调用方自己执行,起到自然限流作用
  • 线程数不是越多越好:CPU 密集型任务线程数 ≈ CPU 核数;IO 密集型可按公式 cpu_count * (1 + wait_time / cpu_time) 估算,但务必压测验证
  • 记得给线程池命名(通过 ThreadFactory),否则堆里全是 pool-1-thread-1,排查问题时根本分不清是谁的池子

CompletableFuture 链式调用中的隐形阻塞点

CompletableFuture 看似异步,但一不小心就写出“伪异步”代码。最典型的是在 thenApplythenAccept 里调用阻塞 API,或者没指定异步执行器,导致默认用 ForkJoinPool.commonPool(),而它会被任意业务代码拖垮。

立即学习Java免费学习笔记(深入)”;

  • 所有可能耗时的操作,必须显式指定线程池:supplyAsync(() -> heavyWork(), executor)thenApplyAsync(..., executor)
  • 慎用 join()get():它们会阻塞当前线程,在 Web 容器(如 Tomcat)中可能导致线程池耗尽
  • 异常处理别漏掉:exceptionally() 只捕获上一阶段异常,handle() 才能同时处理结果和异常,推荐优先用后者
  • 多个依赖关系建议用 allOf + thenCombine 组合,但注意 allOf 返回 CompletableFuture<void></void>,需手动 collect 结果,容易写错
CompletableFuture<String> a = CompletableFuture.supplyAsync(() -> callServiceA(), ioExecutor);
CompletableFuture<Integer> b = CompletableFuture.supplyAsync(() -> callServiceB(), ioExecutor);
return CompletableFuture.allOf(a, b)
    .thenApply(v -> {
        // 错误:a.get() 和 b.get() 会阻塞!
        return a.join() + b.join(); // 正确:join 不抛检异常,且语义清晰
    });

高可用不是靠加机器堆出来的,是靠对每个并发原语的副作用足够敏感。最容易被忽略的,其实是那些“看起来很安全”的默认行为——比如 commonPool 的共享性、ConcurrentHashMap 的弱一致性边界、还有拒绝策略背后的服务降级意图。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
如何配置Tomcat环境变量
如何配置Tomcat环境变量

配置Tomcat环境变量需要在系统中添加CATALINA_HOME变量,并将Tomcat的安装路径添加到PATH变量中。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

117

2023.10.26

idea如何集成Tomcat
idea如何集成Tomcat

idea集成Tomcat的步骤:1、添加Tomcat服务器配置;2、配置项目部署;3、运行Tomcat服务器;4、访问项目;5、注意事项;6、关闭Tomcat服务器。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

173

2024.02.23

怎么查看Tomcat源代码
怎么查看Tomcat源代码

查看Tomcat源代码的步骤:1、下载Tomcat源代码;2、在IDEA中导入Tomcat源代码;3、查看源代码;4、理解Tomcat的工作原理;5、参与社区和贡献;6、注意事项;7、持续学习和更新;8、使用工具和插件。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

98

2024.02.23

常见的tomcat漏洞有哪些
常见的tomcat漏洞有哪些

常见的tomcat漏洞有:1、跨站脚本攻击;2、跨站请求伪造;3、目录遍历漏洞;4、缓冲区溢出漏洞;5、配置漏洞;6、第三方组件漏洞。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

170

2024.02.23

tomcat日志乱码怎么解决
tomcat日志乱码怎么解决

tomcat日志乱码的解决办法:1、修改tomcat的日志编码设置;2、检查ide的编码设置;3、检查操作系统的编码设置;4、使用过滤器处理日志;5、检查外部系统的编码设置;6、检查文件编码方式等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

158

2024.02.23

weblogic和tomcat有哪些区别
weblogic和tomcat有哪些区别

weblogic和tomcat的区别:1、功能;2、性能;3、规模;4、价格;5、安全性;6、配置和管理;7、社区支持;8、集成能力;9、升级和更新;10、可靠性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

200

2024.02.23

tomcat和nginx有哪些区别
tomcat和nginx有哪些区别

tomcat和nginx的区别:1、应用领域;2、性能;3、功能;4、配置;5、安全性;6、扩展性;7、部署复杂性;8、社区支持;9、成本;10、日志管理。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

244

2024.02.23

tomcat启动闪退怎么解决
tomcat启动闪退怎么解决

tomcat启动闪退的解决办法:1、检查java环境;2、检查环境变量配置;3、检查端口被占用;4、检查配置文件编码;5、检查启动时需要的配置文件;6、检查相关文件是否丢失;7、检查防火墙和杀毒软件设置。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

169

2024.02.23

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.2万人学习

Java 教程
Java 教程

共578课时 | 80.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号