Java多线程中sleep()方法和wait()方法的区别

目录

具体而言 ,sleep()方法与wait()方法的区别主要表现在以下几个方面:

引申:sleep()方法和yield()方法有什么区别?

常见面试题:

1、利用Thread.wait()同步线程,可以设置超时时间吗?

2、在一个线程中sleep(1000)方法中,将使该线程在多长时间获得对CPU的控制(假设睡眠过程中不会有其他事件唤醒该线程)?


      sleep()是使线程暂停执行一段时间的方法。wait()也是一种使得线程暂停执行的方法,例如,当线程交互时,如果线程对一个同步对象x发出一个wait()调用请求,那么该线程会暂停执行,被调对象进入等待状态该,直到被唤醒或者等待时间超时

具体而言 ,sleep()方法与wait()方法的区别主要表现在以下几个方面:

1.1)原理不同。sleep()方法时Thred的静态方法,是线程用来控制自身流程的,它会使得此线程暂停执行一段时间,而把执行机会让给其他线程,等到计时时间一到,此线程会自动”苏醒“,例如,当线程执行报时功能时,每一秒钟打印一次时间,那么此时就需要在打印的方法前面加上一个sleep()方法,以便于让自己每隔1s执行一次,该过程如同闹钟一样。而wait()方法是Object类的方法,用于线程之间的通信,这个方法会使当前拥有该对象锁的进程等待,直到其他线程调用notify()方法,或者(notifyAll方法)时才会“醒”来。与wait()方法配套的方法还哟有notify()方法和notifyAll()方法。

1.2)对锁的处理机制不同。由于sleep()方法的主要作用是让线程暂停执行一段时间,时间一到则自动恢复,不涉及线程间的通信,因此调用sleep()方法并不会释放锁。而wait()方法则不同,当调用wait方法()后,线程会释放掉它所占用的锁,从而使线程所在对象中的其他synchronized数据可被别的线程使用。举个简单的例子,如果张三拿遥控器看电视的期间,可用sleep()方法每个10分钟调用一次频道,而在这十分钟里,遥控器就在张三手里。

1.3)使用区域不同。由于wait()方法的特殊意义,因此它必须放在同步控制方法或者同步语句块中使用,而sleep()方法则可以放在任何地方使用。

      sleep()方法不会释放“锁标志”,容易导致死锁问题的发生,因一般情况下不推荐使用sleep()方法,而推荐使用wait()方法。

引申:sleep()方法和yield()方法有什么区别?

2.1)sleep()方法给其他线程运行机会时不考虑线程的优先级,因此会给低优先级的线程以运行的机会,而yield()方法只会给相同优先级或者更高优先级别的线程以运行的机会。

2.2)线程执行sleep()方法后会转入阻塞状态,所以,执行sleep()方法的线程在指定的时间内肯定不会被执行,而yield()方法只是使得当前线程重新回到可执行状态,所以执行yield()方法的线程有可能在进入到可执行状态后马上又被执行。

2.3)sleep()方法生命抛出InterruptedException,而yield()方法没有声明任何异常。

2.4)sleep()方法比yield()方法(跟操作系统相关)具有更好的可移植性。

常见面试题:

1、利用Thread.wait()同步线程,可以设置超时时间吗?

A、可以 B、不可以

答案:A。可以设置超时,函数原型为wait(long timeout)和wait(long timeout,int nanos) timeout代表最长的等待时间,单位为ms(毫秒);nanos代表额外的等待时间,单位为ns(奈秒,十亿分之一秒)。

2、在一个线程中sleep(1000)方法中,将使该线程在多长时间获得对CPU的控制(假设睡眠过程中不会有其他事件唤醒该线程)?

A、正好1000ms B、少于1000ms C、大于等于1000ms D、不一定

答案:C。sleep()方法指定的时间为线程不会运行的最短时间。当睡眠时间结束后,线程会返回到可运行状态,不是运行状态,还需要等待CPU调度运行。因此,sleep()方法不能保证该线程睡眠到期后就开始执行。

本文节选自《Java程序员面试宝典》的多线程篇一节

手打码字的,有错别字的话海涵~

猜你喜欢

转载自blog.csdn.net/lps12345666/article/details/130252642