1. 程式人生 > >跟著例項學習java多執行緒3-synchronized的多種寫法有何區別?

跟著例項學習java多執行緒3-synchronized的多種寫法有何區別?

同步程式碼塊是一種有效實現操作原子性的方法,上一章我們講了一些同步的原子操作的基礎。

現在我們回憶一下上一章的兩個問題。

1:不同的synchronized的寫法有什麼區別,又該怎麼寫建立執行緒的程式碼呢?

以class例項物件作為鎖的寫法

寫法1

package com.home.thread;

/**
 * @author gaoxu
 * 
 */
public class SafeThread {
	@safe
	public  void testPrint(){
		 synchronized(SafeThread.class){
			System.out.println("Enter testPrint method !");
			try {
				Thread.sleep(2000);
			} catch (InterruptedException e) {
		
				e.printStackTrace();
			}	
			System.out.println("Exit testPrint method !");
		}
	}
	
}

寫法2

package com.home.thread;

/**
 * @author gaoxu
 * 
 */
public class SafeThread {
		
	public static   synchronized  void testPrint(){
			System.out.println("Enter testPrint method !");
			try {
				Thread.sleep(2000);
			} catch (InterruptedException e) {
		
				e.printStackTrace();
			}	
			System.out.println("Exit testPrint method !");
	}
	
}

以上兩種寫法是以class例項物件為鎖的寫法,這兩種寫法的呼叫執行緒該怎麼寫呢?讓我們來看下面的例子

寫法1,建立當前物件例項,並使用物件例項初始化執行緒。

package com.home.thread;

/**
 * @author gaoxu
 * 
 */
public class ThreadStart {
	
	public static void main(String[] para){
		SafeThread safe = new SafeThread();
		for(int i=0;i<3;i++){
			ThreadRead1 t1 = new ThreadRead1(safe);
			t1.start();
		}
	}
}
package com.home.thread;

/**
 * @author gaoxu
 *
 */
public class ThreadRead1 extends Thread{
	SafeThread safe = null;
	public ThreadRead1(){
	}
	public ThreadRead1(SafeThread o){
		safe = o;
	}

	public void run()
	{
		safe.testPrint();
	}

}


寫法2,可以線上程中建立類例項。

package com.home.thread;

/**
 * @author gaoxu
 * 
 */
public class ThreadStart {
	
	public static void main(String[] para){
		for(int i=0;i<3;i++){
			ThreadRead1 t1 = new ThreadRead1();
			t1.start();
		}
	}
}</span>
<span style="font-size:14px;">package com.home.thread;

/**
 * @author gaoxu
 *
 */
public class ThreadRead1 extends Thread{
	SafeThread safe = null;
	public ThreadRead1(){
	}
	
	public void run()
	{
		safe = new SafeThread();
		safe.testPrint();
	}

}


這兩總寫法可以起到相同的作用,都可以實現原子的操作,實現同步互斥的呼叫。

建立內部同步程式碼塊,以當前例項物件作為鎖的物件。

寫法1

package com.home.thread;

/**
 * @author gaoxu
 * 
 */
public class SafeThread {
		public   synchronized  void testPrint(){
			System.out.println("Enter testPrint method !");
			try {
				Thread.sleep(2000);
			} catch (InterruptedException e) {
		
				e.printStackTrace();
			}	
			System.out.println("Exit testPrint method !");
	}
	
}
package com.home.thread;

/**
 * @author gaoxu
 * 
 */
public class SafeThread {
		public  void testPrint(){
		synchronized(this){
			System.out.println("Enter testPrint method !");
			try {
				Thread.sleep(2000);
			} catch (InterruptedException e) {
		
				e.printStackTrace();
			}	
			System.out.println("Exit testPrint method !");
		}

	
}
	
}<span style="font-size:14px;">
</span>
package com.home.thread;

/**
 * @author gaoxu
 * 
 */
public class SafeThread {
		Object a = new Object();

		
	public   void testPrint(){
		synchronized(a){
			System.out.println("Enter testPrint method !");
			try {
				Thread.sleep(2000);
			} catch (InterruptedException e) {
		
				e.printStackTrace();
			}	
			System.out.println("Exit testPrint method !");
		}

	
}
	
}

這三種寫法都是以類的當前例項物件作為鎖物件,所以執行緒呼叫寫法如下;

package com.home.thread;

/**
 * @author gaoxu
 * 
 */
public class ThreadStart {
	
	public static void main(String[] para){
		SafeThread safe = new SafeThread();//當前例項物件
		for(int i=0;i<3;i++){
			ThreadRead1 t1 = new ThreadRead1(safe);
			t1.start();
		}
	}
}
package com.home.thread;

/**
 * @author gaoxu
 *
 */
public class ThreadRead1 extends Thread{
	SafeThread safe = null;
	public ThreadRead1(){
	}
	public ThreadRead1(SafeThread o){
		safe = o;
	}

	public void run()
	{
		safe.testPrint();
	}

}

下面讓我們來看一下以class物件和當前物件的區別:

    類.class和static synchronized是對該類所有例項物件枷鎖。

    synchronized(this),synchronized,synchronized(object)都是對類的當前例項物件加鎖。

具體說明:

      synchronized是對類的當前例項進行加鎖,防止其他執行緒同時訪問該類的當前例項的所有synchronized塊。static synchronized是控制類的所有例項的訪問了,static synchronized是限制執行緒同時訪問jvm中該類的所有例項對應的程式碼快。在類中某方法或某程式碼塊中有 synchronized,那麼在生成一個該類例項後,該類也就有一個監視塊,放置執行緒併發訪問該例項synchronized保護塊,這個保護塊只對當前例項有效,而static synchronized則是所有該類的例項公用一個監視塊,放置執行緒併發訪問該類所有例項的保護塊,synchronized相當於 this.synchronized,而
static synchronized相當於Something.synchronized。

2:死鎖、活躍性問題都是怎麼產生的。

後續章節我們重點討論。

今天的問題是:

1:執行緒安全除了原子操作還有什麼需要注意的。

2:如何確定自己需要實現一個執行緒安全類。