一、线程通信目标
1、线程通信的目标是使线程间能够互相发送信号
2、线程通信使线程能够等待其他线程的信号
二、几种方式
1、通过共享对象
2、忙等待
线程 B 运行在一个循环里,以等待信号 (不释放cpu)
3、wait,notify和notifyAll
wait会使线程进入睡眠或者非运行状态,释放cpu使用权;
线程必须在同步块里调用 wait()或者 notify();
当一个线程调用一个对象的 notify()方法,正在等待该对象的所有线程中将有一个线程被唤醒并允许执行(校注:这个将被唤醒的线程是随机的,不可以指定唤醒哪个线程)。同时也提供了一个 notifyAll()方法来唤醒正在等待一个给定对象的所有线程;
三、信号丢失
notify()和 notifyAll()方法不会保存调用它们的方法,因为当这两个方法被调用时,有可能没有线程处于等待状态。通知信号过后便丢弃了。因此,如果一个线程先于被通知线程调用 wait()前调用了 notify(),等待的线程将错过这个信号。
为了避免丢失信号,必须把它们保存在信号类里。
四、假唤醒
线程有可能在没有调用过 notify()和 notifyAll()的情况下醒来。这就是所谓的假唤醒(spurious wakeups)。
为了防止假唤醒,保存信号的成员变量将在一个 while 循环里接受检查,而不是在 if 表达式里。这样的一个 while 循环叫做自旋锁(校注:这种做法要慎重,目前的 JVM 实现自旋会消耗 CPU,如果长时间不调用 doNotify 方法,doWait 方法会一直自旋,CPU 会消耗太大)。被唤醒的线程会自旋直到自旋锁(while 循环)里的条件变为 false。
五、不要在字符串常量或全局对象中调用 wait()
在空字符串作为锁的同步块(或者其他常量字符串)里调用 wait()和 notify()产生的问题是,JVM/编译器内部会把常量字符串转换成同一个对象;
应该使用对应唯一的对象