文章目录
Thread的几个重要方法
线程通知与等待wait()/notify()/notifyAll()
Object类是所有类的父类,鉴于继承机制,Java把所有类都需要的方法放在了Object中,现在要用到的通知与等待系列函数也在其中。
wait()函数
当一个线程调用一个**共享变量的wait()**方法时,该调用线程会被阻塞挂起。
wait()方法是属于共享变量的!!!
线程想要申请一个共享变量,则需要等待共享变量空闲或者满足某种程序员需要的条件才可以申请到。如果共享变量是非空闲的状态或不满足条件,那么需要给共享变量加监视锁(监视到什么时候空闲或满足某种条件)并释放自己在当前变量上的其他锁以便别的线程使用(在其他变量上的锁是不会被释放的),等待其他占用线程释放该变量并唤醒该锁。
挂起线程什么时候被取下来开始返回进入运行状态呢?满足如下条件之一即可:
- 其他线程调用了该共享对象的notify()/notifyAll()方法唤醒。
- 其他线程调用了该线程的interrupt()方法中断,该线程抛出InterruptedException异常返回(需要进行异常处理)。
- 等待超时
获取监视器锁
synchronized同步代码块
使用该共享变量作为参数:
synchronized(共享变量) {
//do something
}
共享变量方法,方法使用synchronized修饰
synchronized void add(int a, int b) {
//do something
}
虚假唤醒
在不满足上述三个唤醒条件的情况下,也可能会从挂起变为运行状态,这就是所谓的虚假唤醒。为了避免这种情况发生,应该不停测试线程被唤醒的条件是否满足,不满足就继续等待(while)。
synchronized (obj) {
while(条件不满足) {
obj.wait();
}
}
举例:消费者&生产者
唤醒函数
notify()函数
一个线程在调用共享对象的notify()方法后,会唤醒一个在共享变量上wait等待的线程,如果有多个线程在等待,具体唤醒哪一个则是随机的。
被唤醒线程不会马上返回,必须在获取了共享对象的监视器锁才能返回(唤醒他的线程释放了他的监视器锁之后,但被唤线程不一定能获得此锁,得竞争)。
同时,只有当前进程获取到监视器锁才能调用notify()方法,否则会抛异常。
notifyAll()函数
唤醒共享变量上的所有wait挂起的线程。
等待线程执行终止join
有些场景需要等待某些事情完成后才能继续往下执行:多个线程加载资源等…
wait()等待通知方法是Object类中的方法
join()方法是Thread类直接提供的,无参,返回值void。
亲测:join()方法是阻塞,使得主线程等待join()线程执行完毕再往下执行。
CountDownLatch
睡眠sleep
sleep方法是Thread类中的一个静态方法。
如果线程调用sleep方法,则线程会暂时让出指定时间的执行权,在这期间不参与CPU调度,但是所持监视器资源(锁等)不会让出。
睡眠时间到后,函数正常返回,线程处于就绪状态,参与CPU调度,获得CPU资源后可以继续运行。
如果睡眠期间,其他线程调用interrupt()中断该线程,则该线程在调用sleep处抛出异常InterrupetException并返回。
让出CPU执行权yield()
yield()是Thread类中的一个静态方法。
一个线程调用yield方法,向线程调度器表明线程请求让出自己的CPU使用。但是线程调度器可以无条件忽略请求。
调用yield方法,当前线程让出CPU使用权,然后处于就绪状态。
一般较少是用这个方法,这个方法在调试或者测试是可以帮助复现由于并发竞争条件导致的问题。