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

Java线程的sleep和wait的区别

       在Java中,Thread.sleep()Object.wait() 都可以让线程暂停执行,但是它们的作用机制和使用场景是不同的。下面是这两个方法的主要区别:

Thread.sleep(long millis, int nanos)

  • 参数millis 是毫秒数,nanos 是额外的纳秒数(0到999,999之间)。
  • 行为:当前线程将暂停执行至少指定的毫秒数加上纳秒数。实际暂停时间可能会更长,因为系统调度或其他因素可能会导致额外的延迟。
  • 锁的影响Thread.sleep() 不会影响线程所持有的锁。如果线程在调用 Thread.sleep() 之前持有了某个对象的锁,那么即使在线程暂停期间,这个锁也不会被释放。
  • 异常处理:如果另一个线程中断了正在睡眠的线程,那么会抛出 InterruptedException,并且清除中断状态。因此,在调用 Thread.sleep() 后应该总是捕获 InterruptedException 并进行适当的处理。
  • 应用场景:适用于简单的定时任务,比如每隔一段时间执行一次某些操作。
  • 用法

              基本用法:使用 Thread.sleep(millis) 或 Thread.sleep(millis, nanos) 让线程暂停指定的时间。 

               异常处理:必须捕获 InterruptedException,因为如果在 sleep 期间线程被中断,就会抛出这个异常。处理中断异常的一种常见做法是重新设置中断标志,然后退出当前操作。

示例
try {
    Thread.sleep(1000); // 使当前线程暂停1秒钟
} catch (InterruptedException e) {
    // 重设中断状态
    Thread.currentThread().interrupt();
    System.out.println("Thread was interrupted during sleep.");
}

Object.wait()

  • 参数wait() 可以接受三个不同的签名:
    • void wait(): 无限期等待,直到被其他线程唤醒。
    • void wait(long timeout): 等待指定毫秒数后自动醒来。
    • void wait(long timeout, int nanos): 等待指定的毫秒数加纳秒数后自动醒来。
  • 行为:当前线程必须首先获取调用 wait() 方法的对象的锁。调用 wait() 后,线程会释放该对象的锁,并进入等待队列。直到其他线程调用了同一个对象的 notify() 或 notifyAll() 方法,或者指定了超时时间并已过期,线程才会重新获取锁并继续执行。
  • 锁的影响:与 Thread.sleep() 不同,Object.wait() 会导致线程释放它所持有的对象锁,这样其他线程就有机会获得该锁并执行。
  • 异常处理:同样地,如果线程在等待期间被中断,wait() 方法会抛出 InterruptedException
  • 应用场景:适用于需要线程间协作的场合,比如当一个线程需要等待某个条件成立(如缓冲区中有数据可用)才能继续执行时。
  • 用法

               同步上下文Object.wait() 必须在 synchronized 块内调用,因为它需要当前线程拥有该对象的锁。

               唤醒机制Object.notify() 只会随机唤醒一个等待中的线程,而 Object.notifyAll() 会唤醒所有等待中的线程。

                异常处理:同样需要捕获 InterruptedException

示例
synchronized (sharedResource) {
    while (!condition) { // condition 是一个布尔值,表示等待的条件
        try {
            sharedResource.wait(); // 线程等待,直到被唤醒
        } catch (InterruptedException e) {
            // 重设中断状态
            Thread.currentThread().interrupt();
            System.out.println("Thread was interrupted during wait.");
            return;
        }
    }
    // 执行条件满足后的操作
}

深入理解

  • 锁的管理Thread.sleep() 保持锁,而 Object.wait() 释放锁。这对于线程间协作非常重要,因为 Object.wait() 允许其他线程访问共享资源,而 Thread.sleep() 不允许。
  • 线程间通信Object.wait() 和 Object.notify()/Object.notifyAll() 提供了一种方式让线程间进行通信。一个线程可以等待特定条件,而另一个线程可以在条件满足时通知等待的线程。
  • 死锁预防:由于 Object.wait() 释放锁,所以它可以用来避免死锁。相反,Thread.sleep() 保持锁,如果使用不当,这可能会导致死锁。

注意事项

  • Thread.sleep() 和 Object.wait() 都是静态方法,但 Object.wait() 必须在 synchronized 上下文中调用。
  • 使用 Object.wait() 和 Object.notify()/Object.notifyAll() 时,一定要确保这些方法是在同一个对象上被调用的,否则线程可能永远不会被唤醒。
  • 在处理 InterruptedException 时,通常的做法是恢复中断状态(如果需要的话),以便调用栈中的其他方法能够正确地响应中断。

       你可以根据你的具体需求选择合适的方法来控制线程的行为。如果你只是想简单地暂停线程而不涉及线程间的交互,那么 Thread.sleep() 就足够了;如果你需要实现更复杂的线程间同步逻辑,那么应该考虑使用 Object.wait() 以及相应的 notify()notifyAll() 方法。


http://www.kler.cn/a/393253.html

相关文章:

  • 漏洞扫描:网络安全的 “体检” 与 “防护指南”
  • tryhackme-Cyber Security 101-Linux Shells(linux命令框)
  • NLP 中文拼写检测纠正论文 C-LLM Learn to CSC Errors Character by Character
  • SQL子查询和having实例
  • Glossary 词汇表中英文对照
  • 初始 ShellJS:一个 Node.js 命令行工具集合
  • 力扣 LeetCode 349. 两个数组的交集(Day3:哈希表)
  • 一文理解吸收《红黑树》的精华
  • AI生成字幕模型whisper介绍与使用
  • 软件开发中的 Pull Request 工作流:逐步指南
  • springboot读取modbus数据
  • 《AI 使生活更美好》
  • 【go从零单排】gin+gorm理解及实现CRUD
  • 机器学习系列----KNN分类
  • 【Linux网络】Linux网络编程套接字,UDP与TCP
  • PCB安全电气间距
  • Python网络爬虫与数据采集实战——网络协议与HTTP
  • linux命令详解,存储管理相关
  • 排序算法 -堆排序
  • SQL面试题——奔驰SQL面试题 车辆在不同驾驶模式下的时间
  • 学Linux的第八天
  • 数字IC实践项目(10)—基于System Verilog的DDR4 Model/Tb 及基础Verification IP的设计与验证(付费项目)
  • uniapp 上传 base64 图片
  • ubuntu22.04上手指南(更新阿里源、安装ssh、安装chrome、设置固定IP、安装搜狗输入法)
  • 【二叉搜素树】——LeetCode二叉树问题集锦:6个实用题目和解题思路
  • 除了 Postman,还有什么好用的 API 测试工具吗