0

0

如何利用Java的Semaphore实现接口限流_并发数控制实战

P粉602998670

P粉602998670

发布时间:2026-03-07 11:58:03

|

167人浏览过

|

来源于php中文网

原创

semaphore 是并发数控制而非限流,它限制同时执行的线程数,不控制单位时间请求数(qps);适用于同步阻塞服务保护下游资源,不适用于异步模型或短耗时接口。

如何利用java的semaphore实现接口限流_并发数控制实战

为什么 Semaphore 不是“限流”而是“并发数控制”

很多人一看到接口限流就直接上 Semaphore,结果压测时发现 QPS 没降、超时却变多——根本原因是搞混了概念:Semaphore 控制的是**同时执行的线程数**,不是单位时间请求数(QPS)。它适合保护下游资源不被并发打垮(比如数据库连接池、本地缓存写入),但对防刷、削峰这类真正“限流”场景效果有限。

典型误用:给一个 HTTP 接口方法加 semaphore.acquire(),却没考虑异步 Servlet、线程复用(如 Tomcat 的 NIO 线程池)、或 Spring WebFlux 的非阻塞模型——这时 acquire() 可能阻塞在 IO 线程上,反而拖垮整个线程池。

  • 适用场景:同步阻塞式服务(如传统 Spring MVC + Tomcat 同步模式)
  • 不适用场景:WebFlux、Vert.x、gRPC 异步服务;或接口本身耗时极短(Semaphore 的 CAS 开销反而成瓶颈
  • 关键判断点:你的请求处理逻辑是否「必然在同一线程内完成」?如果不是,别硬套 Semaphore

Semaphore 初始化必须用 false(非公平模式)

默认构造函数 new Semaphore(permits) 实际等价于 new Semaphore(permits, false),但很多人会下意识写成 new Semaphore(permits, true)——这是个隐蔽陷阱。公平模式会维护一个 FIFO 队列,所有 acquire() 请求排队等待,吞吐量直接掉 30%~50%,且在高并发下容易出现“长尾获取”:某个请求卡在队列中间,等几十毫秒才拿到许可,用户体验断崖式下跌。

真实压测数据:100 并发下,非公平模式平均获取延迟

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

光子AI
光子AI

AI电商服饰商拍平台

下载
  • 永远显式声明 new Semaphore(10, false),别依赖默认
  • 如果业务真需要“先到先得”,优先考虑用 RateLimiter + 时间窗口,而不是靠 Semaphore 公平性
  • 注意:公平性开关只影响 acquire(),不影响 tryAcquire(long, TimeUnit) 的超时行为

必须配合 tryAcquire 和超时,不能无脑 acquire

acquire() 是无限等待,一旦后端响应变慢或 release() 被遗漏(比如异常没 catch 到),许可就会永久泄漏,Semaphore 迅速变成“空转红灯”——所有新请求卡死,监控看不出来,排查要翻日志找哪条链路没 release

正确姿势是:所有 acquire 必须带超时,且 release 放在 finally 块里。

if (!semaphore.tryAcquire(100, TimeUnit.MILLISECONDS)) {
    throw new RuntimeException("Too many requests");
}
try {
    // 执行业务
} finally {
    semaphore.release();
}
  • 超时值建议设为接口 SLO 的 1/3(比如 P95 是 300ms,这里设 100ms)
  • 不要用 tryAcquire() 无参版本——它不保证原子性,在极端竞争下可能返回 true 却实际没拿到许可
  • 如果业务逻辑里有嵌套调用、异步分支,确保每个分支都覆盖 release,否则必漏

和 Spring AOP 结合时,切点必须选对方法签名

@Around 切 Controller 方法时,如果切的是 public 方法但实际被 CGLIB 代理(比如类没实现接口),AOP 可能失效;更常见的是切了 HandlerMethodHttpServletRequest,结果 Semaphore 在 Filter 层就被拿走了,Controller 根本没机会执行。

最稳的方式是切具体业务 Service 方法,且该方法必须是 public + 由 Spring 容器管理的 bean。

  • 错误示范:@Pointcut("execution(* com.example.controller..*.*(..))") —— Controller 层太薄,异常可能抛在拦截器之后
  • 推荐切点:@Pointcut("execution(* com.example.service..*Service.*(..))")
  • 务必检查 @EnableAspectJAutoProxy(proxyTargetClass = true) 是否开启,否则 JDK 动态代理对没有接口的类无效
  • 切记:AOP 代理对象里的 this 不是 Spring Bean,所以不能在切面里直接调 this.method(),否则绕过代理,Semaphore 彻底失效

复杂点在于,如果你的业务方法里又调了另一个加了同样切面的方法,得确认 Semaphore 是按“每方法独立计数”还是“整条调用链共用”——前者要每个方法配独立 Semaphore,后者得用 ThreadLocal + 可重入设计,但那就不是原生 Semaphore 能解决的了。

热门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服务器。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

172

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、第三方组件漏洞。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

169

2024.02.23

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

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

157

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

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

1

2026.03.06

热门下载

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

精品课程

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

共23课时 | 4.2万人学习

C# 教程
C# 教程

共94课时 | 10.8万人学习

Java 教程
Java 教程

共578课时 | 77.8万人学习

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

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