百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

沉睡与守望:Java线程中sleep 和 wait 的区别

yund56 2025-05-02 20:54 20 浏览

在Java的平行世界里,每个线程都像一位忙碌的工人。有人选择“躺平”(sleep),定好闹钟准时复工;有人选择“守望”(wait),交出工作证等待同伴唤醒——这两种暂停方式看似相似,却藏着截然不同的生存哲学。今天,我们就走进这个微观世界,拆解sleep与wait的隐秘边界。

基础概念:两种暂停的基因差异

1. 出身不同,使命各异

  • sleep:生于Thread贵族(静态方法),无需看人脸色,随时随地喊停就停。就像程序员小王在工位上闭目养神,但手里还攥着会议室的门卡(不释放锁)
  • wait:来自Object草根(实例方法),必须依附于某个对象。如同快递员小李在小区门口等待包裹,必须把快递车钥匙交给保安(释放锁)才能暂时离开。

2. 代码里的众生相

Java
// sleep的任性时刻(任意场景调用)
Thread.sleep(2000); // 躺平2秒

// wait的生存法则(必须在同步块中)
synchronized(lock) {
    lock.wait(); // 交出钥匙等待唤醒
}

七维度解剖差异本质

对比维度

sleep

wait

暂停原理

定时休眠,闹钟叫醒

被动等待,需他人唤醒

锁处理

紧握锁如同溺水者抓稻草

优雅放手成就他人

唤醒机制

时间到自动满血复活

需notify/notifyAll发信号弹

异常处理

可能被InterruptedException打断美梦

同左,但更易被中断

线程状态

TIMED_WAITING(带期限等待)

WAITING(无限期)/TIMED_WAITING

使用场景

单线程节奏控制

多线程协作交响曲

性能影响

可能引发死锁危机

更优的资源利用率


致命误区:那些年我们踩过的坑

1. 同步块里的沉睡陷阱

Java
synchronized(lock) {
    Thread.sleep(5000); // 危险!其他线程干着急
}

后果:就像把整个办公室唯一的热水壶锁进抽屉午睡,同事们只能干瞪眼

2. 野生wait引发的后果

Java
public void rogueWait() {
    lock.wait(); // 报错:没在同步块里!
}

教训:没拿到工作证就想交钥匙?系统保安会直接把你赶出场

3. 中断处理的温柔陷阱

Java
try {
    Thread.sleep(1000);
} catch (InterruptedException e) {
    // 吞掉异常如同吃掉警示灯
}

正确姿势:至少记录日志,理想情况恢复中断状态

生存指南:高手的选择与替代方案

1. 选择之道:四象限决策法

  • 定时任务:优先选ScheduledExecutorService代替裸sleep
  • 资源等待:必须用wait搭建生产者-消费者管道
  • 简单延迟:sleep够用但记得处理异常
  • 精准控制:LockSupport.parkNanos()更专业

2. 代码美化

Java
// 原始版
Thread.sleep(1500); 

// 优雅版
TimeUnit.SECONDS.sleep(1);
TimeUnit.MILLISECONDS.sleep(500);

3. 防死锁三原则

  • 锁的持有时间控制在毫秒级
  • 避免嵌套获取多个锁
  • 同步块内绝不调用外部方法

实战:外卖系统的生死时速

场景:饿了么骑手接单系统,3个骑手竞争5份订单。

错误示范

Java
synchronized(orderPool) {
    while(!hasOrder()) {
        Thread.sleep(1000); // 所有骑手集体沉睡
    }
    takeOrder();
}

结果:订单池明明有货,骑手们却都在睡大觉!

正确姿势

Java
synchronized(orderPool) {
    while(!hasOrder()) {
        orderPool.wait(); // 释放锁让系统能添加新订单
    }
    takeOrder();
}

// 系统添加新订单时
synchronized(orderPool) {
    addOrder();
    orderPool.notifyAll();
}

效果:骑手们有序轮班,系统吞吐量提升300%


随着虚拟线程(Project Loom)的来临,传统的暂停方式正在经历变革:

  1. 结构化并发:通过Scope自动管理生命周期
  2. 纤程调度:更轻量级的暂停恢复机制
  3. 响应式编程:CompletableFuture等异步方案崛起

但无论技术如何演进,理解sleep与wait的底层逻辑,永远是构建稳健并发系统的基石。

相关推荐

豆包编程能力升级:支持HTML代码实时预览、交互

IT之家3月19日消息,IT之家从豆包官方获悉,豆包宣布AI编程功能迎来三项升级,包括HTML预览、Python运行、生成完整项目。据介绍,目前豆包支持HTML代码实时预览和交互...

1898款游戏!80、90回忆杀,重温旧梦,快速搭建中文DOS游戏服务

本内容来源于@什么值得买APP,观点仅代表作者本人|作者:羊刀仙大家好,我是羊刀仙。本期来介绍一个特别情怀向的游戏项目:chinese-dos-games。这套包含1898款经典中文DOS游戏的合集...

利用 SVG 文件内的 HTML 代码进行网络钓鱼攻击

随着时间的推移,网络钓鱼攻击的技术越来越精妙,旨在欺骗用户并规避安全措施。攻击者会使用欺骗性的URL重定向策略,例如将恶意网站地址附加到看似安全的链接后,在PDF中嵌入链接,以及发送HTML...

aardio + AI 大模型自动编写 Python 代码、网页前端代码的经验与技巧

在AI时代,老式的编程习惯完全被颠覆。原来可能要一大堆插件或工具辛苦堆出来的程序,现在只要把AI调教好了就行。aardio支持调用十几种编程语言,这很适合发挥AI大模型的优势。对于AI...

用AI制作游戏就是如此简单!

很多人不知道如何利用AI提高效率,不知道AI能帮我们做什么,其实可以让我们实现很多自己根本不懂的领域取得直观体验,比如利用DS或者豆包,输入“我想做一个简单的单机俄罗斯方块游戏”,AI会给出phtho...

不会写代码?教你用DeepSeek 三步做出小游戏

如今,借助人工智能技术,哪怕你完全看不懂代码,也能通过DeepSeek制作出属于自己的网页版大鱼吃小鱼游戏。接下来,就为大家详细介绍制作过程。第一步、向DeepSeek描述需求为何要做网页版的...

《暗黑1》被移植成网页游戏 可在浏览器上玩了

《暗黑1》,这款1996年发售的“鼠标杀手”砍杀游戏,现在可以在浏览器上玩了。国外专注暴雪游戏的Rivsoft分享了一个《暗黑1》的共享版本,该版本只包含地下城的头2个地区和三个角色职业中的一个。不...

网页代码过滤 轻松获取专辑目录

通过过滤网页代码,可以将网页上显示不全的长文件名列表完整地提取出来。我有一个含有75个视频文件的《中医诊断学》课件,文件名是以01.RMVB、02.RMVB……75.RMVB这种格式命名的。我希望能找...

IDEA 2021首个大版本发布,Java开发者感动哭了(附新亮点演示)

工欲善其事,必先利其器!就在不久之前,Java领域的开发神器IntelliJIDEA终于迎来2021年的一个重要的大版本更新:IntelliJIDEA2021.1。现如今大量的Java开发者深度...

View Source:在 iOS 上轻松查看网页源代码

在移动互联网时代,移动端的应用和web体验都尤为重要,在PC上有很多web前端工具可以选择,而在移动端貌似就少之又少了,在NEXT出现的ViewSource能帮你在iOS上查看...

当我们《寻找房祖名》,我们能找到什么?

游戏葡萄原创专稿,未经允许请勿转载柯震东,因为在九把刀电影《那些年我们追过的女孩》中饰演男主角柯景腾而走红的台湾影星,在昨天被爆出了和著名演员成龙之子房祖名吸毒被抓的丑闻,一时间相关讨论席卷社交网络。...

多用途游戏娱乐新闻网站HTML5模板

Retnews是一个响应式的HTML新闻,博客,杂志网站模板,可以使用这套前端模板简约很多设计的工作。模板有许多特性适合流行的主题商业、时尚,游戏,娱乐,生活方式、体育、科技、政治、旅行、天气、视频等...

简约好看的个人引导页HTML源码下载

源码介绍一款非常简约好看的个人引导页HTML源码,非常适合个人主页以及个人导航使用,纯HTML不需要数据库,上传服务器即可使用!...

教你如何在微信公共平台上插入小游戏(图文教程)

很多玩微信公共平台的朋友都想在公共平台上面插入几个小游戏,用来跟用户之间互动,这里花生来分享一下如何在微信公共平台上插入游戏,以及如何制作html5微信小游戏。首先是找游戏,总共有三个方法,本人比较倾...

html5重力感应剖析附源码

下面是测试html5重力感应的demohttp://bbs.qietu.com/html/zhongli/http://www.qietu.com/html/f2/qqqianbao/demo2是切图...