wait和sleep都会暂停当前线程,对于cpu资源来说,两者都会使cpu将时间片分配给别的线程。
1 用法
sleep是Thread
类的方法,wait是Object
类的方法。
sleep可以在任何地方使用,而wait只能在同步方法或者同步块中使用。
1 | try { |
1 | Object mon = ...; |
1 | synchronized (mon) { |
2 资源锁释放
sleep()方法正在执行的线程主动让出CPU(然后CPU就可以去执行其他任务),在sleep指定时间后CPU再回到该线程继续往下执行(注意:sleep方法只让出了CPU,而并不会释放同步资源锁!!!);wait()方法则是指当前线程让自己暂时退让出同步资源锁,以便其他正在等待该资源的线程得到该资源进而运行,只有调用了notify()方法,之前调用wait()的线程才会解除wait状态,可以去参与竞争同步资源锁,进而得到执行。(注意:notify的作用相当于叫醒睡着的人,而并不会给他分配任务,就是说notify只是让之前调用wait的线程有权利重新参与线程的调度);
如线程状态图所示,当线程处于running
状态,此时线程正在执行,当该线程执行到Thread.sleep()
,之后处于阻塞
状态,当sleep()
时间结束后,此时该线程处于runnable
状态,如果之后被cpu选中,则可以继续执行。
如果线程处于running
状态,之后开始执行o.wait()
方法,此时处于等待队列,在被别的线程唤醒之后,处于锁池中,之后还需要和别的线程竞争锁,获得锁之后,才转为runnable
状态,如果之后被cpu选中,则可以继续执行。
下面这段代码具体展示了相关的内容。
1 | package com.exercise.MultitThread; |
执行结果
1 | 主线程 睡眠10ms |
我们在主线程中启动了两个线程,Thread1、Thread2。在Thread1中,通过synchronized的同步方法块中,使用wait()
方法,需要注意锁的是类对象
。在Thread2中,同样通过synchronized的同步方法块锁住类对象
,之后调用了notify()
方法,接着调用了sleep()
方法。通过执行结果,我们可以发现Thread1遇到wait()
方法之后,释放了类锁。因为Thread2使用的是同一把类锁,只有Thread1释放类锁,Thread2才能得到。Thread2在调用notify()
方法后,此时Thread1 处于竞争锁的状态。Thread2调用了sleep()
方法,在执行完毕后,Thread1才继续执行,说明Thread1在Thread2执行完毕之后才获得了锁。因此,在Thread1执行sleep()
方法时,没有释放锁。
3 参考
- https://blog.csdn.net/u012050154/article/details/50903326 Java中sleep()与wait()区别
- https://www.zhihu.com/question/23328075 java sleep和wait的区别的疑惑?