目录
  • 区别一:语法使用不同
  • 区别二:所属类不同
  • 区别三:唤醒方式不同
  • 区别四:释放锁资源不同
    • sleep 不释放锁
    • wait 释放锁
  • 区别五:线程进入状态不同
    • 总结

      前言:

      sleep 方法和 wait 方法都是用来将线程进入休眠状态的,并且 sleep 和 wait 方法都可以响应 interrupt 中断,也就是线程在休眠的过程中,如果收到中断信号,都可以进行响应并中断,且都可以抛出 InterruptedException 异常,那 sleep 和 wait 有什么区别呢?接下来,我们一起来看。

      区别一:语法使用不同

      wait 方法必须配合synchronized 一起使用,不然在运行时就会抛出 IllegalMonitorStateException 的异常,如下代码所示: 

      java中sleep方法和wait方法的五个区别

       初看代码好像没啥问题,编译器也没报错,然而当我们运行以上程序时就会发生如下错误: 

      java中sleep方法和wait方法的五个区别

       而 sleep 可以单独使用,无需配合 synchronized 一起使用。

      区别二:所属类不同

      wait 方法属于 Object 类的方法,而 sleep 属于 Thread 类的方法,如下图所示: 

      java中sleep方法和wait方法的五个区别

      区别三:唤醒方式不同

      sleep 方法必须要传递一个超时时间的参数,且过了超时时间之后,线程会自动唤醒。而 wait 方法可以不传递任何参数,不传递任何参数时表示永久休眠,直到另一个线程调用了 notify 或 notifyAll 之后,休眠的线程才能被唤醒。也就是说 sleep 方法具有主动唤醒功能,而不传递任何参数的 wait 方法只能被动的被唤醒

      区别四:释放锁资源不同

      wait 方法会主动的释放锁,而 sleep 方法则不会。接下来我们使用代码的方式来演示一下二者的区别。

      sleep 不释放锁

      接下来使用 sleep 是线程休眠 2s,然后在另一个线程中尝试获取公共锁,如果能够获取到锁,则说明 sleep 在休眠时会释放锁,反之则说明不会释放锁,实现代码如下:

      public static void main(String[] args) throws InterruptedException {
          Object lock = new Object();
          new Thread(() -> {
              synchronized (lock) {
                  System.out.println("新线程获取到锁:" + LocalDateTime.now());
                  try {
                      // 休眠 2s
                      Thread.sleep(2000);
                      System.out.println("新线程获释放锁:" + LocalDateTime.now());
                  } catch (InterruptedException e) {
                      e.printStackTrace();
                  }
              }
          }).start();
          // 等新线程先获得锁
          Thread.sleep(200);
          System.out.println("主线程尝试获取锁:" + LocalDateTime.now());
          // 在新线程休眠之后,尝试获取锁
          synchronized (lock) {
              System.out.println("主线程获取到锁:" + LocalDateTime.now());
          }
      }

      以上代码的执行结果如下图所示: 

      java中sleep方法和wait方法的五个区别

       从上述结果可以看出,在调用了 sleep 之后,在主线程里尝试获取锁却没有成功,只有 sleep 执行完之后释放了锁,主线程才正常的得到了锁,这说明 sleep 在休眠时并不会释放锁。

      wait 释放锁

      接下来使用同样的方式,将 sleep 替换成 wait,在线程休眠之后,在另一个线程中尝试获取锁,实现代码如下:

      public static void main(String[] args) throws InterruptedException {
          Object lock = new Object();
          new Thread(() -> {
              synchronized (lock) {
                  System.out.println("新线程获取到锁:" + LocalDateTime.now());
                  try {
                      // 休眠 2s
                      lock.wait(2000);
                      System.out.println("新线程获释放锁:" + LocalDateTime.now());
                  } catch (InterruptedException e) {
                      e.printStackTrace();
                  }
              }
          }).start();
          // 等新线程先获得锁
          Thread.sleep(200);
          System.out.println("主线程尝试获取锁:" + LocalDateTime.now());
          // 在新线程休眠之后,尝试获取锁
          synchronized (lock) {
              System.out.println("主线程获取到锁:" + LocalDateTime.now());
          }
      }

      以上代码的执行结果如下图所示: 

      java中sleep方法和wait方法的五个区别

       从上述结果可以看出,当调用了 wait 之后,主线程立马尝试获取锁成功了,这就说明 wait 休眠时是释放锁的。

      区别五:线程进入状态不同

      调用 sleep 方法线程会进入 TIMED_WAITING 有时限等待状态,而调用无参数的 wait 方法,线程会进入 WAITING 无时限等待状态 

      代码演示:

      public static void main(String[] args) throws InterruptedException {
          Object lock = new Object();
          Thread t1 = new Thread(() -> {
              synchronized (lock) {
                  try {
                      // 休眠 2s
                      lock.wait();
                  } catch (InterruptedException e) {
                      e.printStackTrace();
                  }
              }
          });
          t1.start();
      
          Thread t2 = new Thread(() -> {
              try {
                  Thread.sleep(2000);
              } catch (InterruptedException e) {
                  e.printStackTrace();
              }
          });
          t2.start();
      
          Thread.sleep(200);
          System.out.println("wait() 之后进入状态:" + t1.getState());
          System.out.println("sleep(2000) 之后进入状态:" + t2.getState());
      
      }

      以上代码的执行结果如下:

      java中sleep方法和wait方法的五个区别

      总结

      sleep 和 wait 都可以让线程进入休眠状态,并且它们都可以响应 interrupt 中断,但二者的区别主要体现在:语法使用不同、所属类不同、唤醒方式不同、释放锁不同和线程进入的状态不同。​

      声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。