0

0

Java里的Semaphore信号量怎么用_控制并发访问资源数量

P粉602998670

P粉602998670

发布时间:2026-03-09 15:18:12

|

435人浏览过

|

来源于php中文网

原创

new semaphore(n) 的 n 应根据资源真实承载能力设定,如数据库连接池为10则设10;需预留缓冲、避免动态修改、用tryacquire防阻塞、release必须放finally、非公平模式性能更优。

java里的semaphore信号量怎么用_控制并发访问资源数量

new Semaphore(n) 的 n 设多少才合理

信号量的许可数 n 不是拍脑袋定的,它直接对应「你允许同时访问该资源的线程上限」。比如数据库连接池最大 10 个连接,那 new Semaphore(10) 才匹配;设成 20 就可能触发连接超时异常,设成 5 则人为压低吞吐。

常见错误是把 n 和线程池大小混为一谈——线程池有 20 个线程,不代表资源能撑住 20 并发。关键看资源本身瓶颈:文件句柄、第三方 API QPS 限制、内存敏感的缓存计算等。

  • 查清下游资源的真实承载能力(如 Redis 官方建议单实例不要超过 10k 并发连接,但实际业务接口可能 50 并发就响应变慢)
  • 预留缓冲:如果压测发现 8 并发时 P99 延迟开始上扬,别设 new Semaphore(8),设 76 更稳妥
  • 避免动态改 n:构造后调用 release(int)acquire(int) 可能绕过预期控制,除非你明确需要可变配额

acquire() 阻塞时线程中断会被忽略吗

会。默认 acquire() 不响应中断,线程卡在那儿,Thread.interrupt() 白叫。这在超时控制或服务优雅停机时非常危险——你发了中断,线程却还在等许可,停不掉。

正确做法是用 acquireUninterruptibly()(不响应中断,适合必须拿到许可的场景),或更常用的是 tryAcquire(long, TimeUnit) 配合超时判断。

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

Stable Diffusion Online
Stable Diffusion Online

基于Stable Diffusion搭建的AI绘图工具

下载
  • if (!sem.tryAcquire(3, TimeUnit.SECONDS)) { throw new RuntimeException("获取许可超时"); }
  • 别写 sem.acquire(); // 危险!无超时、不响应中断
  • 如果真要用阻塞版,至少包一层 try-catch (InterruptedException e),并在 catch 里做清理(如还原状态、记录日志),然后重新抛出或返回

release() 忘记调用会导致什么

许可证永远不归还,availablePermits() 持续减少,最终所有线程卡死在 acquire()。这不是内存泄漏,但效果类似——系统逐渐失去响应能力,且极难排查,因为堆栈里只看到一堆 WAITING 状态的线程在等 Semaphore

典型场景:带 try-finally 的写法被简化,或者异步回调里漏了 release()

  • 必须把 release() 放进 finally 块,哪怕只是单行逻辑
  • 避免在 lambda 或 CompletableFuture 回调中直接调用 release(),容易因异常跳过——先捕获再显式 release
  • 调试时可定期打点:System.out.println("permits left: " + sem.availablePermits());,上线前删掉

公平模式(fair = true)有什么实际影响

new Semaphore(n, true) 后,等待线程按 FIFO 顺序获取许可。听起来很理想?但代价是每次 acquire()release() 都要走队列操作,性能比非公平模式低 10%–30%,尤其在高并发争抢时。

除非你明确需要「先到先得」语义(比如任务排队执行不能乱序),否则默认非公平更合适。JVM 默认就是非公平,也符合多数服务对吞吐优先的取舍。

  • 公平模式不会防止饥饿,只是让等待队列有序;如果持续有新线程快速 acquire/release,老线程仍可能长期等不到
  • 测试时开启公平模式可能掩盖竞争 bug——因为调度太「规矩」,反而看不出真实压力下的行为
  • 线上环境几乎不用公平模式,除非业务逻辑强制要求顺序性(如支付扣减必须严格按请求时间)

最麻烦的不是怎么写对,而是许可证和业务逻辑的生命周期怎么对齐——比如一次 HTTP 请求开了 3 个 DB 查询,该用 1 个许可还是 3 个?这里没标准答案,得看你保护的是连接池本身,还是单次查询的资源消耗。细节错一点,压测时才暴露,但那时已经晚了。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

846

2023.08.22

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

990

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

607

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

314

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

214

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

192

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

61

2026.01.05

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

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

59

2026.03.06

热门下载

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

精品课程

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

共23课时 | 4.2万人学习

C# 教程
C# 教程

共94课时 | 11万人学习

Java 教程
Java 教程

共578课时 | 79.4万人学习

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

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