0

0

详解线程池中的Worker内部类_理解线程复用的底层循环机制

P粉602998670

P粉602998670

发布时间:2026-02-26 13:02:03

|

436人浏览过

|

来源于php中文网

原创

worker是线程池中封装thread并实现runnable的内部类,通过while循环复用执行任务,利用aqs状态控制中断,其生命周期由gettask()超时策略决定,不可外部直接操作。

详解线程池中的worker内部类_理解线程复用的底层循环机制

Worker 是什么?它不是 Thread,而是 Runnable 的“包装器”

线程池里真正干活的不是裸露的 Thread,而是一个叫 Worker 的内部类——它实现了 Runnable,同时持有一个 Thread 实例。你调用 execute() 提交任务时,线程池不会直接把任务塞给 Thread.run(),而是让 Worker 自己去循环取任务、执行、再取下一个。

关键在于:Workerrun() 方法里根本没写死只执行一次,而是调用了 runWorker(this),后者是个 while 循环体:

final void runWorker(Worker w) {
  Runnable task = w.firstTask;
  while (task != null || (task = getTask()) != null) {
    task.run(); // 这里才是你传进来的 Runnable
    task = null;
  }
}
  • firstTask 是创建 Worker 时指定的初始任务(可能为 null),用于支持预热或立即执行
  • 循环中 getTask() 才是复用的核心:它从 workQueue 阻塞或超时取任务,取不到就等,取到就执行
  • 一旦 getTask() 返回 null,while 结束,Worker 退出,对应线程自然终止

getTask() 怎么决定线程该不该回收?看两个条件

getTask() 不是无脑阻塞,它会根据当前线程数和配置动态选择等待策略,直接决定非核心线程是否被回收:

  • 如果 allowCoreThreadTimeOut == true,所有线程都按超时逻辑走
  • 否则,仅当 workerCount > corePoolSize(即当前是非核心线程)时,才启用超时等待
  • 超时等待用的是 workQueue.poll(keepAliveTime, unit);不超时则用 workQueue.take()(永久阻塞)

也就是说:核心线程默认永不超时,哪怕队列空着也一直卡在 take() 上;而非核心线程在空闲满 keepAliveTime 后,poll() 返回 null,整个 Worker 就退出了

常见踩坑点:keepAliveTime 设得太小(比如 10ms),又频繁提交低频任务,会导致非核心线程刚建好就回收,反复创建销毁,反而失去复用意义。

企奶奶
企奶奶

一款专注于企业信息查询的智能大模型,企奶奶查企业,像聊天一样简单。

下载

为什么 Worker 要继承 AQS?锁在哪用?

Worker 继承 AbstractQueuedSynchronizer(AQS)不是为了并发控制任务,而是为了**安全中断线程**——它用 AQS 的 state 做线程运行状态标记(-1 初始化、0 运行中、1 已完成),并在执行任务前后加锁/解锁。

  • 执行前调用 lock(),防止外部在任务运行中调用 interrupt()
  • 执行后 unlock(),允许后续中断操作生效
  • 真正中断线程靠的是 thread.interrupt(),但必须等当前任务跑完、且 getTask() 再次进入阻塞前才生效

所以你看到的 shutdownNow() 并不能立刻杀死正在执行的任务,它只是:① 设置线程池状态为 STOP;② 对所有 Worker.thread 调用 interrupt();③ 清空队列。任务是否响应中断,完全取决于你自己的 Runnable 是否检查 Thread.interrupted()

别把 Worker 当普通线程对象去操作

有人试图拿到 Worker 实例后手动调用 run() 或修改 firstTask,这是错的——Worker 是线程池私有实现,没有公开 API,也不保证线程安全。

  • 不能 new Worker(...) 然后 start(),它的 thread 字段由线程池通过 ThreadFactory 创建并绑定
  • 不能在外部保存 Worker 引用并期望它长期有效;线程退出后,该对象会被 GC,且不再关联任何线程
  • 监控线程池运行状态,请用 ThreadPoolExecutor 提供的 getActiveCount()getCompletedTaskCount() 等方法,而不是遍历内部 workers Set

真正需要定制行为的地方,是替换 ThreadFactory(比如统一命名、设置守护属性)或重写 beforeExecute()/afterExecute() 钩子方法——这些才是开放、稳定、可维护的扩展点。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

248

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

906

2024.03.01

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

104

2023.09.25

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

721

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

95

2025.12.01

Golang 实际项目案例:从需求到上线
Golang 实际项目案例:从需求到上线

《Golang 实际项目案例:从需求到上线》以真实业务场景为主线,完整覆盖需求分析、架构设计、模块拆分、编码实现、性能优化与部署上线全过程,强调工程规范与实践决策,帮助开发者打通从技术实现到系统交付的关键路径,提升独立完成 Go 项目的综合能力。

1

2026.02.26

Golang Web 开发路线:构建高效后端服务
Golang Web 开发路线:构建高效后端服务

《Golang Web 开发路线:构建高效后端服务》围绕 Go 在后端领域的工程实践,系统讲解 Web 框架选型、路由设计、中间件机制、数据库访问与接口规范,结合高并发与可维护性思维,逐步构建稳定、高性能、易扩展的后端服务体系,帮助开发者形成完整的 Go Web 架构能力。

3

2026.02.26

Golang 并发编程专题:掌握多核时代的核心技能
Golang 并发编程专题:掌握多核时代的核心技能

《Golang 并发编程专题:掌握多核时代的核心技能》系统讲解 Go 在并发领域的设计哲学与实践方法,深入剖析 goroutine、channel、调度模型与并发安全机制,结合真实场景与性能思维,帮助开发者构建高吞吐、低延迟、可扩展的并发程序,全面提升多核时代的工程能力。

5

2026.02.26

batoto漫画官网入口与网页版访问指南
batoto漫画官网入口与网页版访问指南

本专题系统整理batoto漫画官方网站最新可用入口,涵盖最新官网地址、网页版登录页面及防走失访问方式说明,帮助用户快速找到batoto漫画官方平台,稳定在线阅读各类漫画内容。

353

2026.02.25

热门下载

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

精品课程

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

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