当前位置: 首页 > 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

相关文章:

  • 向日葵软件Windows系统连接苹果系统(MacOS)的无反应问题解决办法
  • INQUIRE:一个包含五百万张自然世界图像,涵盖10,000个不同物种的专为专家级文本到图像检索任务设计的新型基准数据集。
  • 时间管理的三个痛点
  • python高效处理大数据:将Excel10万数据分批插入MySQL数据库的实战代码
  • 【C++】一种针对代码的连续条件检查方案,累计布尔结果
  • vue2或vue3的name属性有什么作用?
  • 力扣 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 测试工具吗