Java中join()的核心原理是调用线程在目标线程对象上执行wait(),循环检查isAlive()直至其终止,JVM在线程结束时自动调用notifyAll()唤醒等待线程;带超时的join通过循环减法精确控制等待时间,避免虚假唤醒。

Java 中 join() 的核心原理是:调用线程在目标线程对象上执行 wait(),持续等待直到目标线程终止(进入 TERMINATED 状态),期间通过循环检查 isAlive() 并配合同步锁与等待唤醒机制实现阻塞与恢复。
本质是基于 wait/notify 的对象级等待
join 方法被声明为 synchronized,意味着它锁的是被调用的线程对象本身(比如 t.join() 锁的是 t 对象)。内部逻辑是:
- 只要目标线程还活着(
t.isAlive() == true),就调用t.wait(0)让当前线程进入 WAITING 状态 - 当目标线程自然结束(run 方法执行完)或异常退出时,JVM 会自动在其对象上调用
notifyAll() - 被阻塞的线程被唤醒后,重新检查
isAlive();若已死亡,跳出循环,join 方法返回
带超时的 join 是循环减法 + 精确等待
比如 t.join(3000) 不是一次性等 3 秒,而是:
- 记录起始时间
base - 每次
wait(delay)后重新计算剩余等待时间:delay = 3000 - 已耗时 - 如果剩余时间 ≤ 0,直接退出循环,不等了
- 这种设计能避免虚假唤醒(spurious wakeup)导致误判
不是“抢占”也不是“调度干预”
join 不影响线程优先级,也不让出 CPU 给目标线程。它只是挂起调用方,不参与线程调度决策。目标线程是否运行、何时运行,完全由 JVM 和操作系统调度器决定 —— join 只负责“守着它死”,不负责“帮它活”。
在原金领办公系统上增加的客户管理系统,其中包括客户管理,文档管理,表格管理,分超级管理员(总经办),管理员(部门经理),普通用户(员工),强大的检索功能,各种条件检索,普通用户删除的客户资料,超级管理员能在回收站中恢复,有效的保证了员工离职使客户资源丢失问题。超级管理员:ao密码:123456
立即学习“Java免费学习笔记(深入)”;
和 sleep、yield 的关键区别
对比常见线程控制方法:
-
sleep(ms):当前线程休眠指定毫秒,不释放锁,不依赖其他线程状态 -
yield():建议调度器让出 CPU,但无实际保证,也不等待任何线程 -
join():明确等待另一个线程终结,依赖其生命周期,且必须在目标线程start()之后调用才有效
基本上就这些。理解 join,关键是抓住“在目标线程对象上 wait,靠 JVM 自动 notify”这个底层动作,而不是把它想象成某种魔法同步指令。










