等待/通知机制
-
什么是等待通知机制?
在单线程的编程中,要执行的操作需要满足一定的条件下才能执行,可以把这个操作放在if语句块中。
在多线程编程中,可能A线程的条件,没有满足只是暂时的,稍后其他的线程B可能更新条件使得A线程的条件得到满足。可以将A线程暂停,直到它的条件得到满足后再将A线程唤醒。
-
等待/通知机制的实现
Object类中的wait()方法可以使执行当前代码的线程等待,暂停执行,直到接到通知或被中断为止。
//1.waitit()方法只能在同步代码块中由锁对象调用
//2.调用wait()方法,当前线程会释放锁
//伪代码
synchronized(锁对象){
while(条件不成立){
//通过锁对象调用wait()方法暂停线程,会释放锁对象
锁对象.wait();
}
//线程的条件满足了继续向下执行
}
Object类的notify()可以唤醒线程,该方法也必须在同步代码块由锁对象调用,没有使用锁对象调用wait()
notify()会抛出IIegaIMonitorStateExeption异常。如果有多个等待的线程,notify()方法只能唤醒其中的一个。在同步代码块中调用notify()后,并不会立即释放锁对象,需要等当前同步代码块执行完后才会释放锁对象,一般将notify()方法放在同步代码块的最后。
//伪代码
synchronized(所对象){
//执行修改保护条件的代码
//唤醒其他线程
锁对象.notify();
}
-
notify()方法后不会立即释放锁对象
-
interrupt()方法会中断wait()
当线程处于wait等待状态时,调用线程对象的interrupt()方法会中断线程的等待状态,会产生InterruptedException异常
-
notify()与notifyAll()
notify()一次只能唤醒一个线程,如果有多个等待的线程,只能随机唤醒其中的一个;想要唤醒所有的等待线程,需要调用notifyAll().
-
wait(long)的使用
wait(long)带有long类型参数的wait()等待,如果在参数指定的时间内没有被唤醒,超时后会自动唤醒。
-
通知过早
线程wait等待后可以调用notify唤醒线程,如果唤醒的过早,在等待之前就调用了,会打乱程序正常的逻辑。
-
wait等待的条件发生了变化
在使用wait/notify模式时,注意wait条件发生了变化,也可能会造成程序逻辑的混乱
生产者消费者模式
在java中,负责产生数据的模块是生产者,负责使用数据的模块是消费者。解决数据的平衡问题,即先有数据才能使用,没有数
据时,消费者需要等待。
通过管道实现线程间的通信
在java.io包中的PipeStream管道流用于在线程之间传送数据。
一个线程发送数据到输出管道,另外一个线程从输入管道中读取数据。
相关的类包括:PipedlnputStream和PipedOutputStream (字节流) 和 PipedWriter (字符流)。
ThreadLocal的使用
除了控制资源的访问之外,还可以通过增加资源来保证线程安全。ThreadLocal主要解决为每个线程绑定自己的值。
- 使用ThreadLocal 为每个线程指定自己的对象
- ThreadLocal 初始值 :(1).定义一个ThreadLocal的子类 (2).写initialValue方法,设置初始值 (这样第一次调用get()就不会返回null)