1. 程式人生 > 其它 >多執行緒訪問Synchronized同步方法的8種使用場景

多執行緒訪問Synchronized同步方法的8種使用場景

技術標籤:【Java-併發程式設計】

目錄

1、場景一:兩個執行緒同時訪問同一個物件的同步方法

2、場景二:兩個執行緒同時訪問兩個物件的同步方法

3、場景三:兩個執行緒同時訪問(一個或兩個)物件的靜態同步方法

4、場景四:兩個執行緒分別同時訪問(一個或兩個)物件的同步方法和非同步方法

5、場景五:兩個執行緒訪問同一個物件中的同步方法,同步方法又呼叫一個非同步方法

6、場景六:兩個執行緒同時訪問同一個物件的不同的同步方法

7、場景七:兩個執行緒分別同時訪問靜態synchronized和非靜態synchronized方法

8、場景八:同步方法丟擲異常後,JVM會自動釋放鎖的情況

總結


本文將介紹7種同步方法的訪問場景,我們來看看著8種情況下,多執行緒訪問同步方法是否還是執行緒安全的。這些場景是多執行緒程式設計中經常遇到的,而且也是面試時高頻被問到的問題,所以不管是理論還是實踐,這些都是多執行緒場景必須要掌握的場景。

1、場景一:兩個執行緒同時訪問同一個物件的同步方法

分析:這種情況是經典的物件鎖中的方法鎖,兩個執行緒爭奪同一個物件鎖,所以會相互等待,是執行緒安全的。

兩個執行緒同時訪問同一個物件的同步方法,是執行緒安全的。
1

案例:通過同步程式碼塊鎖,實現兩個執行緒對同一個全域性變數count,各自執行1萬次count++,驗證結果是否等於2萬,而不會出現小於2萬的情況。

完整程式碼實現:

public class SynchronizeCodeBlockLock implements Runnable {
	private static SynchronizeCodeBlockLock instance = new SynchronizeCodeBlockLock();
	private static int count = 0;

	@Override
	public void run() {
		method();
	}

	private void method() {
		// 關鍵:同步程式碼塊的方式,操作同一變數,達到執行緒安全的效果
		synchronized (this) {
			System.out.println("執行緒名:" + Thread.currentThread().getName() + ",執行開始");
			for (int i = 0; i < 10000; i++) {
				count++;
			}
			System.out.println("執行緒:" + Thread.currentThread().getName() + ",執行結束");
		}
	}

	public static void main(String[] args) {
		Thread thread1 = new Thread(instance);
		Thread thread2 = new Thread(instance);
		thread1.start();
		thread2.start();
		while (thread1.isAlive() || thread2.isAlive()) {
			// 若有執行緒如果還在活動,則不執行下一步(等同於thread.join()方法)
		}
		System.out.println("期待結果:20000,實際結果:" + count);
	}


}

執行結果:

執行緒名:Thread-0,執行開始
執行緒:Thread-0,執行結束
執行緒名:Thread-1,執行開始
執行緒:Thread-1,執行結束
期待結果:20000,實際結果:20000

分析執行結果:

我們發現使用了synchronized關鍵字後,執行緒Thread-0先執行,等到其結束後,Thread-1才開始執行。如果不使用synchronized關鍵字,執行結果可能會是Thread-0和Thread-1幾乎同時執行,幾乎同時執行結束。這就說明使用了synchronized關鍵字後,將多個執行緒的並行,變為了序列。

2、場景二:兩個執行緒同時訪問兩個物件的同步方法

這種場景就是物件鎖失效的場景,原因出在訪問的是兩個物件的同步方法,那麼這兩個執行緒分別持有的兩個執行緒的鎖,所以是互相不會受限的。加鎖的目的是為了讓多個執行緒競爭同一把鎖,而這種情況多個執行緒之間不再競爭同一把鎖,而是分別持有一把鎖,所以我們的結論是:

兩個執行緒同時訪問兩個物件的同步方法,是執行緒不安全的。
1

程式碼驗證:

publicclassCondition2implementsRunnable{
//建立兩個不同的物件
staticCondition2instance1=newCondition2();
staticCondition2instance2=newCondition2();

@Override
publicvoidrun(){
method();
}

privatesynchronizedvoidmethod(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",執行結束");
}

publicstaticvoidmain(String[]args){
Threadthread1=newThread(instance1);
Threadthread2=newThread(instance2);
thread1.start();
thread2.start();
while(thread1.isAlive()||thread2.isAlive()){
}
System.out.println("測試結束");
}
}
123456789101112131415161718192021222324252627282930

執行結果:

兩個執行緒是並行執行的,所以執行緒不安全。

執行緒名:Thread-0,執行開始
執行緒名:Thread-1,執行開始
執行緒:Thread-0,執行結束
執行緒:Thread-1,執行結束
測試結束
12345

程式碼分析:

「問題在此:」兩個執行緒(thread1、thread2),訪問兩個物件(instance1、instance2)的同步方法(method()),兩個執行緒都有各自的鎖,不能形成兩個執行緒競爭一把鎖的局勢,所以這時,synchronized修飾的方法method()和不用synchronized修飾的效果一樣(不信去把synchronized關鍵字去掉,執行結果一樣),所以此時的method()只是個普通方法。

「如何解決這個問題:」若要使鎖生效,只需將method()方法用static修飾,這樣就形成了類鎖,多個例項(instance1、instance2)共同競爭一把類鎖,就可以使兩個執行緒序列執行了。這也就是下一個場景要講的內容。

3、場景三:兩個執行緒同時訪問(一個或兩個)物件的靜態同步方法

這個場景解決的是場景二中出現的執行緒不安全問題,即用類鎖實現:

兩個執行緒同時訪問(一個或兩個)物件的靜態同步方法,是執行緒安全的。
1

本例僅展示方法鎖,在控制執行緒序列執行的示例。方法鎖保證執行緒安全的效果,跟同步程式碼塊是一致的。

public class MethodLock implements Runnable {

	private static MethodLock instance = new MethodLock();

	@Override
	public void run() {
		method();
	}

	//關鍵:synchronized可以保證此方法被順序執行,執行緒1執行完4秒鐘後,執行緒2再執行4秒。不加synchronized,執行緒1和執行緒2將同時執行
	private synchronized void method() {
		System.out.println("執行緒:" + Thread.currentThread().getName() + ",執行開始");
		try {
			//模擬執行一段操作,耗時4秒鐘
			Thread.sleep(4000);
			System.out.println("執行緒:" + Thread.currentThread().getName() + ",執行結束");
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
	}

	public static void main(String[] args) {
		// 模擬:同一個物件下,兩個執行緒,同步執行一個方法(序列執行則為執行緒安全,並行執行,則為執行緒不安全,)
		Thread thread1 = new Thread(instance);
		Thread thread2 = new Thread(instance);
		thread1.start();
		thread2.start();
		while (thread1.isAlive() || thread2.isAlive()) {

		}
		System.out.println("測試結束");
	}
}

執行結果:

執行緒:Thread-0,執行開始
執行緒:Thread-0,執行結束
執行緒:Thread-1,執行開始
執行緒:Thread-1,執行結束
測試結束

發現執行結果中,多個執行緒也是序列執行的,效果跟同步程式碼塊鎖是一致的。

4、場景四:兩個執行緒分別同時訪問(一個或兩個)物件的同步方法和非同步方法

這個場景是兩個執行緒其中一個訪問同步方法,另一個訪問非同步方法,此時程式會不會序列執行呢,也就是說是不是執行緒安全的呢?我們可以確定是執行緒不安全的,如果方法不加synchronized都是安全的,那就不需要同步方法了。驗證下我們的結論:

兩個執行緒分別同時訪問(一個或兩個)物件的同步方法和非同步方法,是執行緒不安全的。
1
publicclassCondition4implementsRunnable{

staticCondition4instance=newCondition4();

@Override
publicvoidrun(){
//兩個執行緒訪問同步方法和非同步方法
if(Thread.currentThread().getName().equals("Thread-0")){
//執行緒0,執行同步方法method0()
method0();
}
if(Thread.currentThread().getName().equals("Thread-1")){
//執行緒1,執行非同步方法method1()
method1();
}
}

//同步方法
privatesynchronizedvoidmethod0(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",同步方法,執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",同步方法,執行結束");
}

//普通方法
privatevoidmethod1(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",普通方法,執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",普通方法,執行結束");
}

publicstaticvoidmain(String[]args){
Threadthread1=newThread(instance);
Threadthread2=newThread(instance);
thread1.start();
thread2.start();
while(thread1.isAlive()||thread2.isAlive()){
}
System.out.println("測試結束");
}

}
1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950

執行結果:

兩個執行緒是並行執行的,所以是執行緒不安去的。

執行緒名:Thread-0,同步方法,執行開始
執行緒名:Thread-1,普通方法,執行開始
執行緒:Thread-0,同步方法,執行結束
執行緒:Thread-1,普通方法,執行結束
測試結束
12345

結果分析

「問題在於此:」method1沒有被synchronized修飾,所以不會受到鎖的影響。即便是在同一個物件中,當然在多個例項中,更不會被鎖影響了。結論:

非同步方法不受其它由synchronized修飾的同步方法影響
1

你可能想到一個類似場景:多個執行緒訪問同一個物件中的同步方法,同步方法又呼叫一個非同步方法,這個場景會是執行緒安全的嗎?

5、場景五:兩個執行緒訪問同一個物件中的同步方法,同步方法又呼叫一個非同步方法

我們來實驗下這個場景,用兩個執行緒呼叫同步方法,在同步方法中呼叫普通方法;再用一個執行緒直接呼叫普通方法,看看是否是執行緒安全的?

publicclassCondition8implementsRunnable{

staticCondition8instance=newCondition8();

@Override
publicvoidrun(){
if(Thread.currentThread().getName().equals("Thread-0")){
//直接呼叫普通方法
method2();
}else{
//先呼叫同步方法,在同步方法內呼叫普通方法
method1();
}
}

//同步方法
privatestaticsynchronizedvoidmethod1(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",同步方法,執行開始");
try{
Thread.sleep(2000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",同步方法,執行結束,開始呼叫普通方法");
method2();
}

//普通方法
privatestaticvoidmethod2(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",普通方法,執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",普通方法,執行結束");
}

publicstaticvoidmain(String[]args){
//此執行緒直接呼叫普通方法
Threadthread0=newThread(instance);
//這兩個執行緒直接呼叫同步方法
Threadthread1=newThread(instance);
Threadthread2=newThread(instance);
thread0.start();
thread1.start();
thread2.start();
while(thread0.isAlive()||thread1.isAlive()||thread2.isAlive()){
}
System.out.println("測試結束");
}

}
1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253

執行結果:

執行緒名:Thread-0,普通方法,執行開始
執行緒名:Thread-1,同步方法,執行開始
執行緒:Thread-1,同步方法,執行結束,開始呼叫普通方法
執行緒名:Thread-1,普通方法,執行開始
執行緒:Thread-0,普通方法,執行結束
執行緒:Thread-1,普通方法,執行結束
執行緒名:Thread-2,同步方法,執行開始
執行緒:Thread-2,同步方法,執行結束,開始呼叫普通方法
執行緒名:Thread-2,普通方法,執行開始
執行緒:Thread-2,普通方法,執行結束
測試結束
1234567891011

結果分析:

我們可以看出,普通方法被兩個執行緒並行執行,不是執行緒安全的。這是為什麼呢?

因為如果非同步方法,有任何其他執行緒直接呼叫,而不是僅在呼叫同步方法時,才呼叫非同步方法,此時會出現多個執行緒並行執行非同步方法的情況,執行緒就不安全了。

對於同步方法中呼叫非同步方法時,要想保證執行緒安全,就必須保證非同步方法的入口,僅出現在同步方法中。但這種控制方式不夠優雅,若被不明情況的人直接呼叫非同步方法,就會導致原有的執行緒同步不再安全。所以不推薦大家在專案中這樣使用,但我們要理解這種情況,並且我們要用語義明確的、讓人一看就知道這是同步方法的方式,來處理執行緒安全的問題。

所以,最簡單的方式,是在非同步方法上,也加上synchronized關鍵字,使其變成一個同步方法,這樣就變成了《場景五:兩個執行緒同時訪問同一個物件的不同的同步方法》,這種場景下,大家就很清楚的看到,同一個物件中的兩個同步方法,不管哪個執行緒呼叫,都是執行緒安全的了。

所以結論是:

兩個執行緒訪問同一個物件中的同步方法,同步方法又呼叫一個非同步方法,僅在沒有其他執行緒直接呼叫非同步方法的情況下,是執行緒安全的。若有其他執行緒直接呼叫非同步方法,則是執行緒不安全的。
1

6、場景六:兩個執行緒同時訪問同一個物件的不同的同步方法

這個場景也是在探討物件鎖的作用範圍,物件鎖的作用範圍是物件中的所有同步方法。所以,當訪問同一個物件中的多個同步方法時,結論是:

兩個執行緒同時訪問同一個物件的不同的同步方法時,是執行緒安全的。
1
publicclassCondition5implementsRunnable{
staticCondition5instance=newCondition5();

@Override
publicvoidrun(){
if(Thread.currentThread().getName().equals("Thread-0")){
//執行緒0,執行同步方法method0()
method0();
}
if(Thread.currentThread().getName().equals("Thread-1")){
//執行緒1,執行同步方法method1()
method1();
}
}

privatesynchronizedvoidmethod0(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",同步方法0,執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",同步方法0,執行結束");
}

privatesynchronizedvoidmethod1(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",同步方法1,執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",同步方法1,執行結束");
}

//執行結果:序列
publicstaticvoidmain(String[]args){
Threadthread1=newThread(instance);
Threadthread2=newThread(instance);
thread1.start();
thread2.start();
while(thread1.isAlive()||thread2.isAlive()){
}
System.out.println("測試結束");
}
}
12345678910111213141516171819202122232425262728293031323334353637383940414243444546

執行結果:

是執行緒安全的。

執行緒名:Thread-1,同步方法1,執行開始
執行緒:Thread-1,同步方法1,執行結束
執行緒名:Thread-0,同步方法0,執行開始
執行緒:Thread-0,同步方法0,執行結束
測試結束
12345

結果分析:

兩個方法(method0()和method1())的synchronized修飾符,雖沒有指定鎖物件,但預設鎖物件為this物件為鎖物件, 所以對於同一個例項(instance),兩個執行緒拿到的鎖是同一把鎖,此時同步方法會序列執行。這也是synchronized關鍵字的可重入性的一種體現。

7、場景七:兩個執行緒分別同時訪問靜態synchronized和非靜態synchronized方法

這種場景的本質也是在探討兩個執行緒獲取的是不是同一把鎖的問題。靜態synchronized方法屬於類鎖,鎖物件是(*.class)物件,非靜態synchronized方法屬於物件鎖中的方法鎖,鎖物件是this物件。兩個執行緒拿到的是不同的鎖,自然不會相互影響。結論:

兩個執行緒分別同時訪問靜態synchronized和非靜態synchronized方法,執行緒不安全。
1

程式碼實現:

publicclassCondition6implementsRunnable{
staticCondition6instance=newCondition6();

@Override
publicvoidrun(){
if(Thread.currentThread().getName().equals("Thread-0")){
//執行緒0,執行靜態同步方法method0()
method0();
}
if(Thread.currentThread().getName().equals("Thread-1")){
//執行緒1,執行非靜態同步方法method1()
method1();
}
}

//重點:用static synchronized 修飾的方法,屬於類鎖,鎖物件為(*.class)物件。
privatestaticsynchronizedvoidmethod0(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",靜態同步方法0,執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",靜態同步方法0,執行結束");
}

//重點:synchronized 修飾的方法,屬於方法鎖,鎖物件為(this)物件。
privatesynchronizedvoidmethod1(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",非靜態同步方法1,執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",非靜態同步方法1,執行結束");
}

//執行結果:並行
publicstaticvoidmain(String[]args){
//問題原因:執行緒1的鎖是類鎖(*.class)物件,執行緒2的鎖是方法鎖(this)物件,兩個執行緒的鎖不一樣,自然不會互相影響,所以會並行執行。
Threadthread1=newThread(instance);
Threadthread2=newThread(instance);
thread1.start();
thread2.start();
while(thread1.isAlive()||thread2.isAlive()){
}
System.out.println("測試結束");
}
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748

執行結果:

執行緒名:Thread-0,靜態同步方法0,執行開始
執行緒名:Thread-1,非靜態同步方法1,執行開始
執行緒:Thread-1,非靜態同步方法1,執行結束
執行緒:Thread-0,靜態同步方法0,執行結束
測試結束
12345

8、場景八:同步方法丟擲異常後,JVM會自動釋放鎖的情況

本場景探討的是synchronized釋放鎖的場景:

只有當同步方法執行完或執行時丟擲異常這兩種情況,才會釋放鎖。
1

所以,在一個執行緒的同步方法中出現異常的時候,會釋放鎖,另一個執行緒得到鎖,繼續執行。而不會出現一個執行緒丟擲異常後,另一個執行緒一直等待獲取鎖的情況。這是因為JVM在同步方法丟擲異常的時候,會自動釋放鎖物件。

程式碼實現:

publicclassCondition7implementsRunnable{

privatestaticCondition7instance=newCondition7();

@Override
publicvoidrun(){
if(Thread.currentThread().getName().equals("Thread-0")){
//執行緒0,執行拋異常方法method0()
method0();
}
if(Thread.currentThread().getName().equals("Thread-1")){
//執行緒1,執行正常方法method1()
method1();
}
}

privatesynchronizedvoidmethod0(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
//同步方法中,當丟擲異常時,JVM會自動釋放鎖,不需要手動釋放,其他執行緒即可獲取到該鎖
System.out.println("執行緒名:"+Thread.currentThread().getName()+",丟擲異常,釋放鎖");
thrownewRuntimeException();

}

privatesynchronizedvoidmethod1(){
System.out.println("執行緒名:"+Thread.currentThread().getName()+",執行開始");
try{
Thread.sleep(4000);
}catch(InterruptedExceptione){
e.printStackTrace();
}
System.out.println("執行緒:"+Thread.currentThread().getName()+",執行結束");
}

publicstaticvoidmain(String[]args){
Threadthread1=newThread(instance);
Threadthread2=newThread(instance);
thread1.start();
thread2.start();
while(thread1.isAlive()||thread2.isAlive()){
}
System.out.println("測試結束");
}

}
1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950

執行結果:

執行緒名:Thread-0,執行開始
執行緒名:Thread-0,丟擲異常,釋放鎖
執行緒名:Thread-1,執行開始
Exceptioninthread"Thread-0"java.lang.RuntimeException
atcom.study.synchronize.conditions.Condition7.method0(Condition7.java:34)
atcom.study.synchronize.conditions.Condition7.run(Condition7.java:17)
atjava.lang.Thread.run(Thread.java:748)
執行緒:Thread-1,執行結束
測試結束
123456789

結果分析:

可以看出執行緒還是序列執行的,說明是執行緒安全的。而且出現異常後,不會造成死鎖現象,JVM會自動釋放出現異常執行緒的鎖物件,其他執行緒獲取鎖繼續執行。

總結

本文總結了並用程式碼實現和驗證了synchronized各種使用場景,以及各種場景發生的原因和結論。我們分析的理論基礎都是synchronized關鍵字的鎖物件究竟是誰?多個執行緒之間競爭的是否是同一把鎖?根據這個條件來判斷執行緒是否是安全的。所以,有了這些場景的分析鍛鍊後,我們在以後使用多執行緒程式設計時,也可以通過分析鎖物件的方式,判斷出執行緒是否是安全的,從而避免此類問題的出現。