- 通過共享對象通信
線程間發(fā)送信號的一個簡單方式是在共享對象的變量里設(shè)置信號值。 - 忙等待(Busy Wait)
準(zhǔn)備處理數(shù)據(jù)的線程B正在等待數(shù)據(jù)變?yōu)榭捎谩Q句話說,它在等待線程A的一個信號 - wait(),notify()和notifyAll()
忙等待沒有對運(yùn)行等待線程的CPU進(jìn)行有效的利用,除非平均等待時(shí)間非常短。否則,讓等待線程進(jìn)入睡眠或者非運(yùn)行狀態(tài)更為明智,直到它接收到它等待的信號。
Java有一個內(nèi)建的等待機(jī)制來允許線程在等待信號的時(shí)候變?yōu)榉沁\(yùn)行狀態(tài)。java.lang.Object 類定義了三個方法,wait()、notify()和notifyAll()來實(shí)現(xiàn)這個等待機(jī)制。 - 丟失的信號(Missed Signals)
notify()和notifyAll()方法不會保存調(diào)用它們的方法,因?yàn)楫?dāng)這兩個方法被調(diào)用時(shí),有可能沒有線程處于等待狀態(tài)。通知信號過后便丟棄了。因此,如果一個線程先于被通知線程調(diào)用wait()前調(diào)用了notify(),等待的線程將錯過這個信號。這可能是也可能不是個問題。不過,在某些情況下,這可能使等待線程永遠(yuǎn)在等待,不再醒來,因?yàn)榫€程錯過了喚醒信號。 - 假喚醒
為了防止假喚醒,保存信號的成員變量將在一個while循環(huán)里接受檢查,而不是在if表達(dá)式里。這樣的一個while循環(huán)叫做自旋鎖。 - 多個線程等待相同信號
- 不要在字符串常量或全局對象中調(diào)用wait()
所以:在wait()/notify()機(jī)制中,不要使用全局對象,字符串常量等。應(yīng)該使用對應(yīng)唯一的對象。
● 靜態(tài)變量:線程非安全。
靜態(tài)變量即類變量,位于方法區(qū),為所有對象共享,共享一份內(nèi)存,一旦靜態(tài)變量被修改,其他對象均對修改可見,故線程非安全。
● 實(shí)例變量:單例模式(只有一個對象實(shí)例存在)線程非安全,非單例線程安全。
實(shí)例變量為對象實(shí)例私有,在虛擬機(jī)的堆中分配,若在系統(tǒng)中只存在一個此對象的實(shí)例,在多線程環(huán)境下,“猶如”靜態(tài)變量那樣,被某個線程修改后,其他線程對修改均可見,故線程非安全;如果每個線程執(zhí)行都是在不同的對象中,那對象與對象之間的實(shí)例變量的修改將互不影響,故線程安全。
● 局部變量:線程安全。
每個線程執(zhí)行時(shí)將會把局部變量放在各自棧幀的工作內(nèi)存中,線程間不共享,故不存在線程安全問題。
public class ThreadShareData {
private static int data=0;
private static Map<Thread,Integer> threadData=new HashMap<Thread,Integer>();
public static void main(String[] args) {
for (int i = 0; i < 2; i++) {
new Thread(new Runnable() {
@Override
public void run() {
data = new Random().nextInt() ;
System.out.println(Thread.currentThread().getName() + ":" + data);
threadData.put(Thread.currentThread(),data);
new A().get();
new B().get();
}
}).start();
}
}
static class A{
public void get(){
data=threadData.get(Thread.currentThread());
System.out.println("A--"+Thread.currentThread().getName()+":"+data);
}
}
static class B{
public void get(){
data=threadData.get(Thread.currentThread());
System.out.println("B--"+Thread.currentThread().getName()+":"+data);
}
}
}
輸出結(jié)果:
Thread-1:308764307
Thread-0:499909925
A--Thread-1:499909925
A--Thread-0:499909925
B--Thread-0:499909925
B--Thread-1:499909925
private static int data=0;
private static Map<Thread,Integer> threadData=new HashMap<Thread,Integer>();
threadData是一個HashMap,同一個線程只能放一個data。