当前位置: 首页 > news >正文

Java多线程等待唤醒机制:从synchronized到Lock+Condition

从synchronized到Lock+Condition

  • 前言
  • 1. 先说传统方式:synchronized + wait/notify
  • 2. 重头戏:Lock + Condition(强烈推荐!)
    • 为什么推荐它?
    • 核心组件:
    • 关键规则:
  • 为什么能精确唤醒?
  • 在实际项目中的应用(阻塞队列)
  • 3. 两种方式对比总结
  • 结论:新项目直接用Lock + Condition!老方式只用来理解历史。

前言

今天来聊聊Java多线程里一个超级经典的话题——等待唤醒机制

在多线程编程里,我们经常遇到“线程协作”的场景:比如一个线程生产数据,另一个线程消费数据;或者两个线程需要严格交替执行(像乒乓球一样你一下我一下)。这时候就需要“等待唤醒”:条件不满足时线程自己睡一觉,等条件好了再被叫醒继续干活。

Java提供了两种实现方式:

  • 老派:synchronized + wait()/notify()
  • 新派(推荐):Lock + Condition

今天重点讲Lock + Condition,因为它更灵活、更高效,是现代并发编程的主流(Java并发包里的阻塞队列都是用这个实现的)。我们会用一个简单例子——两个线程交替打印0~9——来一步步讲解。

1. 先说传统方式:synchronized + wait/notify

这是JDK 1.0就有的方式,简单但有局限。

核心规则:

  • 必须在synchronized同步块里调用wait()/notify()
  • wait():当前线程释放锁,进入等待状态(睡大觉)。
  • notify():随机唤醒一个等待线程。
  • notifyAll():唤醒所有等待线程(生产者消费者场景通常用这个,避免“假死”)。

必须用while检查条件(重要!防止伪唤醒)。

简单例子(交替打印):

publicclassSyncWaitNotifyDemo{privatestaticfinal Object lock=newObject();// 共享锁对象privatestaticint num=0;privatestaticboolean isATurn=true;// true: A的回合publicstaticvoidmain(String[]args){newThread(()->{while(num<10){synchronized(lock){while(!isATurn){// 用while防伪唤醒try{lock.wait();// 不是我的回合,释放锁等待}catch(InterruptedException e){e.printStackTrace();}}System.out.println("A打印: "+num++);isATurn=false;lock.notifyAll();// 唤醒所有(安全)}}},"A").start();newThread(()->{while(num<10){synchronized(lock){while(isATurn){try{lock.wait();}catch(InterruptedException e){e.printStackTrace();}}System.out.println("B打印: "+num++);isATurn=true;lock.notifyAll();}}},"B").start();}}

它能工作,但缺点明显:

  • 只有一个等待队列,所有线程混在一起。
    notifyAll()会把所有线程都唤醒(即使不需要),醒来后又发现条件不满足,再wait——浪费性能(叫“惊群效应”)。
    不支持超时、精确唤醒等高级功能。

2. 重头戏:Lock + Condition(强烈推荐!)

从JDK 1.5开始,JUC包(java.util.concurrent)引入了ReentrantLock和Condition,彻底升级了等待唤醒机制。

为什么推荐它?

  • 精确唤醒:一个Lock可以创建多个Condition,每个Condition有独立的等待队列。你可以“只唤醒消费者”或“只唤醒生产者”,不浪费。
  • 功能更强:支持超时等待(awaitNanos)、不可中断等待等。
  • 性能更好:避免惊群,高并发下更快。
  • 灵活:公平锁可选(避免线程饥饿)。

核心组件:

  • ReentrantLock lock = new ReentrantLock();:可重入锁,手动加锁解锁。
  • Condition cond = lock.newCondition();:可以创建多个,每个是一个独立等待队列。
  • cond.await():释放锁,当前线程进入该Condition的等待队列(睡大觉)。
  • cond.signal():只唤醒该队列的一个线程。
  • cond.signalAll():唤醒该队列的所有线程。

关键规则:

  • 必须先lock.lock()获取锁,再操作Condition。
  • unlock()一定要放finally里(防止死锁)。
  • 永远用while检查条件(防伪唤醒)。

完整例子:交替打印0~9(带详细注释)

Javaimport java.util.concurrent.locks.Condition;importjava.util.concurrent.locks.Lock;importjava.util.concurrent.locks.ReentrantLock;publicclassLockConditionDemo{privatestaticfinal Lock lock=newReentrantLock();// 一把可重入锁,所有线程共享privatestaticfinal Condition condA=lock.newCondition();// A线程专属等待队列(约定)privatestaticfinal Condition condB=lock.newCondition();// B线程专属等待队列(约定)privatestaticint num=0;// 当前数字privatestaticboolean isATurn=true;// true: 轮到A(初始让A先)publicstaticvoidmain(String[]args){// A线程:打印偶数newThread(()->{while(num<10){// 直到打印完9lock.lock();// 1. 先拿锁try{while(!isATurn){// 2. 用while检查:不是我的回合就等condA.await();// 释放锁,当前线程(A)进入condA队列睡觉}// 到这说明轮到我了,且重新拿到了锁System.out.println("A打印: "+num);num++;isATurn=false;// 交给BcondB.signal();// 精确唤醒B(只从condB队列拿一个)}catch(InterruptedException e){e.printStackTrace();}finally{lock.unlock();// 3. 一定释放锁!}}},"A").start();// B线程:打印奇数newThread(()->{while(num<10){lock.lock();try{while(isATurn){// 不是我的回合就等condB.await();// B线程进入condB队列}System.out.println("B打印: "+num);num++;isATurn=true;condA.signal();// 精确唤醒A}catch(InterruptedException e){e.printStackTrace();}finally{lock.unlock();}}},"B").start();}}

运行结果(完美交替):
textA打印: 0
B打印: 1
A打印: 2
B打印: 3

B打印: 9

为什么能精确唤醒?

  • 不是Condition“认人”,而是我们约定:A只在condA等,B只在condB等。
    signal()只去对应队列叫醒人,不会吵醒另一边。

在实际项目中的应用(阻塞队列)

  • Java的ArrayBlockingQueue就是用一个Lock + 两个Condition实现的:

  • notEmpty:所有消费者共享的等待队列(队列空时await)。
    notFull:所有生产者共享的等待队列(队列满时await)。
    生产后notEmpty.signal()(只唤醒一个消费者)。
    消费后notFull.signal()(只唤醒一个生产者)。

多生产者/多消费者时,它们共享同一个Condition队列,signal()只唤醒一个就够了——超级高效!

3. 两种方式对比总结

  • 特性synchronized + wait/notifyLock + Condition等待队列只有一个,所有线程混一起可以多个,独立队列(精确唤醒)唤醒方式notify随机,常用notifyAll(惊群)signal精确,只唤醒需要的功能基础支持超时、中断、公平锁等性能一般高并发下更好使用难度简单(自动加解锁)稍复杂(手动unlock)推荐场景简单同步生产者消费者、阻塞队列、高并发

结论:新项目直接用Lock + Condition!老方式只用来理解历史。

  • 希望这篇文章让你对等待唤醒机制不再迷糊~如果你有疑问,或者想看生产者消费者的完整代码,欢迎留言讨论!
    点赞 + 收藏 + 关注,三连支持一下呗~
http://www.cnnetsun.cn/news/100073.html

相关文章:

  • 边缘Agent的Docker监控实践(资源利用率提升90%的秘密)
  • 揭秘Docker Scout漏洞导出功能:如何快速获取镜像安全报告
  • 【云原生Agent资源调度实战】:Docker环境下高效分配CPU与内存的5大黄金法则
  • 增长有毒?流血三闯港股!希迪智驾带病叩钟:115亿市值撑得住“白条狂欢”吗?
  • 多模态Agent性能骤降?可能是Docker网络隔离没做好(附诊断清单)
  • 为什么你的Docker镜像总被攻破?:可能是扫描频率设置错了
  • 背胶条分类识别:基于计算机视觉的修复状态差异检测与质量评估系统
  • 【新】基于SSM的高校实验室管理系统【包括源码+文档+调试】
  • Python 爬虫实战:沪深 300 股票(下)—— 适当进阶!爬取往期批量数据
  • 超声波传感器:无人机低空飞行的“隐形守护者”
  • 我的服务器被被DDOS攻击了
  • Docker MCP 网关协议转换(企业级应用案例深度剖析)
  • 【编程实践】Windows + PySide6 + Matplotlib 绘图时 WinError 32 的完整排查与解决方案
  • 手把手教你实现生产者-消费者模型(条件变量版)
  • Tool-to-Agent_Retrieval:连接工具与智能体的统一检索框架,让大模型多智能体系统更高效
  • 【Matlab】matlab代码实现随机潮流计算
  • 【Agent服务Docker隔离实战】:20年专家揭秘高效环境隔离的5大核心策略
  • WVP-GB28181-Pro视频监控平台实战部署:企业级解决方案深度解析
  • 揭秘机器视觉环形光源:95%的检测难题迎刃而解!
  • 计算机毕设java的水果销售系统 基于Java的水果销售管理系统设计与实现 Java技术驱动的水果销售信息化平台开发
  • 项目的时间线项目从启动到这周 大概是5周的时间10/28-10/31 Week 1项目初始化/需求讨论/设计文档/后端next.js, typescript技术熟悉 项目运行/调试基
  • Vercel AI SDK部署失败?你可能忽略了这4个Docker版本陷阱
  • TabPFN完整指南:如何用AI模型彻底改变表格数据预测
  • Docker Offload任务状态管理深度解析(专家20年实战经验曝光)
  • 企业Agent的Docker安全扫描实战(从0到1构建自动化扫描体系)
  • 10、Linux 文件操作与管理技巧
  • 勒索软件应急响应实战手册:全流程防护与前瞻应对指南
  • 谷歌关停暗网监控工具:2026年安全防护迎来“精准化”转型
  • Pearcleaner Homebrew管理:3步告别复杂命令行操作
  • 用 XinServer 后端平台开发,项目上线只需几天