wait() 函数与 sleep() 函数的区别
关于异常
wait() 函数不需要捕获异常
sleep() 函数必须捕获异常
关于入参
wait() 函数可以传入参数,也可以不传入参数,传入参数就是在参数结束的时间后开始等待,不传入参数就是直接等待。
sleep() 函数必须传入参数,参数就是休眠时间,时间到了就会自动醒来。
关于 API
sleep() 是 Thread 类的函数,导致此线程暂停执行指定时间,给其他线程执行机会,但是依然保持着监控状态,过了指定时间会自动恢复,调用 sleep() 方法不会释放锁对象。当调用 sleep 方法后,当前线程进入阻塞状态。目的是让出 CPU 给其他线程运行的机会。但是由于 sleep() 方法不会释放锁对象,所以在一个同步代码块中调用这个方法后,线程虽然休眠了,但其他线程无法访问它的锁对象。这是因为 sleep() 方法拥有 CPU 的执行权,它可以自动醒来无需唤醒。而当 sleep() 结束指定休眠时间后,这个线程不一定立即执行,因为此时其他线程可能正在运行。
wait() 方法是 Object 类里的方法,当一个线程执行到 wait() 方法时,它就进入到一个和该对象相关的等待池中,同时释放了锁对象,等待期间可以调用里面的同步方法,其他线程可以访问,等待时不拥有 CPU 的执行权,否则其他线程无法获取执行权。当一个线程执行了 wait() 方法后,必须调用 notify 或者 notifyAll 方法才能唤醒,而且是随机唤醒,若是被其他线程抢到了 CPU 执行权,该线程会继续进入等待状态。由于锁对象可以是任意对象,所以 wait() 方法必须定义在 Object 类中,因为 Obeject 类是所有类的基类。
作用范围
sleep() 是静态方法,也就是说它只对当前对象有效。通过 对象名.sleep() 想让该对象线程进入休眠是无效的,它只会让当前线程进入休眠。
wait 、 notify 和 notifyAll 方法只能在同步方法或者同步代码块中使用,而 sleep 方法可以在任何地方使用。
关于 notify 和 notifyAll 详解请查看后文(第九章、第十章)。
调用者的区别
sleep() 方法是让某个线程暂停运行一段时间,其控制范围是由当前线程决定,运行的主动权是由当前线程来控制(拥有 CPU 的执行权)。
wait() 方法是使一个线程进入等待状态,并且释放其所持有的锁对象,notify 方法是通知等待该锁对象的线程重新获得锁对象,然而如果没有获得锁对象,wait 方法和 notify 方法都是没有意义的,因此必须先获得锁对象再对锁对象进行进一步操作于是才要把 wait 方法和 notify 方法写到同步方法和同步代码块中了。所以 wait 、 notify 和 notifyAll 方法要和 synchronized 关键字一起使用。由此可知 wait 和 notify 、 notifyAll 方法是由确定的对象即锁对象来调用的,锁对象就像一个传话的人,他对某个线程说停下来等待,然后对另一个线程说你可以执行了(实质上是被捕获了),这一过程是线程通信。
本质区别
其实两者的区别都是让线程暂停运行一段时间,但本质的区别:
sleep() 是线程的运行状态控制
wait() 是线程间的通信。