1.使用synchronized關鍵字修飾類或者代碼塊;
2.使用Volatile關鍵字修飾變量;
3.在類中加入重入鎖
舉例子:多個線程在處理一個共享變量的時候,就會出現線程安全問題。(相當於多個窗口賣票的操作)
非同步狀態下:
public static void main(String[] args){
Increase increase = new Increase();
int count = 10;
while (count != 0){
new Thread(() -> {
increase.increasementAndPrint();
}).start();
count --;
}
}
static class Increase {
private int i = 0;
void increasementAndPrint() {
System.out.print(i++ + "\n");
}
}
這種情況下可能會導致多個線程輸出的i相同:
0
0
1
2
3
4
5
6
7
8
使用同步:
1.使用synchronized關鍵字
//保證原子性和有序性
static class Increase {
private int i = 0;
synchronized void increasementAndPrint() {
System.out.println(i++);
}
}
2.使用volatile
//不保證原子性(一個被volatile聲明的變量主要有可見性,有序性)
static class Increase {
private volatile int i = 0;
void increasementAndPrint() {
System.out.println(i++);
}
}
volatile關鍵字不一定能夠保證線程安全的問題,其實,在大多數情況下volatile還是可以保證變量的線程安全問題的。所以,在滿足以下兩個條件的情況下,volatile就能保證變量的線程安全問題:
- 運算結果並不依賴變量的當前值,或者能夠確保只有單一的線程修改變量的值。
3.使用重入鎖
//保證原子性和有序性
static class Increase {
private ReentrantLock reentrantLock = new ReentrantLock();
private int i = 0;
void increasementAndPrint() {
reentrantLock.lock();
System.out.println(i++);
reentrantLock.unlock();
}
}

