多個線程之間共享數據的方式探討

1、概述

    一、若是每一個線程執行的代碼相同,可使用同一個Runnable對象,這個Runnable對象中有那個共享數據,例如,買票系統就能夠這麼作。java

/** 
* @Title: TicketSoldThreadTest.java 
* @Package com.lh.threadtest.t7 
* @Description: TODO
* @author Liu 
* @date 2018年1月17日 上午9:58:08 
* @version V1.0 
*/
package com.lh.threadtest.t7;

/** 
* @ClassName: TicketSoldThreadTest 
* @Description: 多個線程之間共享數據的方式探討
* 
* 若是每一個線程執行的代碼相同,可使用同一個Runnable對象,這個Runnable對象中有那個共享數據。
* 例如,買票系統就能夠這麼作
* 
* @author Liu
* @date 2018年1月17日 上午9:58:08 
*  
*/
public class TicketSoldThreadTest {

	/***
	* @Title: main 
	* @Description: TODO
	* @param @param args
	* @return void
	* @throws 
	*/
	public static void main(String[] args) {
		TicketSoldBusiness ticketSoldBusiness = new TicketSoldBusiness();
		for(int i = 0; i < 2; i++){
			new Thread(ticketSoldBusiness).start();
		}
		
	}

}

class TicketSoldBusiness implements Runnable{
	private int count = 10;
	
	private void dec(){
		count--;
		System.out.println(Thread.currentThread().getName() + " count: " + count);
	}

	/* (非 Javadoc) 
	* <p>Title: run</p> 
	* <p>Description: </p>  
	* @see java.lang.Runnable#run() 
	*/
	@Override
	public void run() {
		while(count > 0){
			dec();
		}
	}
	
}

    二、若是每一個線程執行的代碼不一樣,這時候須要用不一樣的Runnable對象,有以下兩種方式來實現這些Runnable對象之間的數據共享:ide

        ① 將共享數據封裝在另一個對象中,而後將這個對象逐一傳遞給各個Runnable對象。每一個線程對共享數據的操做方法也分配到那個對象身上去完成,這樣容易實現針對該數據進行的各個操做的互斥和通訊。this

/** 
* @Title: MultiThreadShareData.java 
* @Package com.lh.threadtest.t7 
* @Description: TODO
* @author Liu 
* @date 2018年1月16日 下午9:45:37 
* @version V1.0 
*/
package com.lh.threadtest.t7;

/** 
* @ClassName: MultiThreadShareData 
* @Description: 多個線程之間共享數據的方式探討
* 
* 題目:
* 	設計4個線程,其中兩個線程每次對j增長1,另外兩個線程對j每次都減小1,寫出程序。
* 
* @author Liu
* @date 2018年1月16日 下午9:45:37 
*  
*/
public class MultiThreadShareData2 {
	
	/***
	* @Title: main 
	* @Description: TODO
	* @param @param args
	* @return void
	* @throws 
	*/
	public static void main(String[] args) {
		Business business = new Business();
		
		for(int i = 0; i< 2; i++){
			new Thread(new Inc(business)).start();
			new Thread(new Dec(business)).start();
		}
	}
}

class Inc implements Runnable{
	private Business business;

	public Inc(Business business){
		this.business = business;
	}

	/* (非 Javadoc) 
	* <p>Title: run</p> 
	* <p>Description: </p>  
	* @see java.lang.Runnable#run() 
	*/
	@Override
	public void run() {
		for(int i = 0; i< 5; i++){
			this.business.inc();
		}
	}
	
}

class Dec implements Runnable{
	private Business business;
	
	public Dec(Business business){
		this.business = business;
	}
	
	/* (非 Javadoc) 
	 * <p>Title: run</p> 
	 * <p>Description: </p>  
	 * @see java.lang.Runnable#run() 
	 */
	@Override
	public void run() {
		for(int i = 0; i< 5; i++){
			this.business.dec();
		}
	}
	
}

class Business{
	private int j = 0;
	
	public synchronized void inc(){
		j++;
		System.out.println(Thread.currentThread().getName() + "中 j = " + j);
	}
	
	public synchronized void dec(){
		j--;
		System.out.println(Thread.currentThread().getName() + "中 j = " + j);
	}
}

        ② 將這些Runnable對象做爲某一個類中的內部類,共享數據做爲這個外部類中的成員變量,每一個線程對共享數據的操做方法也分配給外部類,以便實現對共享數據進行的各個操做的互斥和通訊,做爲內部類的各個Runnable對象調用外部類的這些方法。spa

/** 
* @Title: MultiThreadShareData.java 
* @Package com.lh.threadtest.t7 
* @Description: TODO
* @author Liu 
* @date 2018年1月16日 下午9:45:37 
* @version V1.0 
*/
package com.lh.threadtest.t7;

/** 
* @ClassName: MultiThreadShareData 
* @Description: 多個線程之間共享數據的方式探討
* 
* 題目:
* 	設計4個線程,其中兩個線程每次對j增長1,另外兩個線程對j每次都減小1,寫出程序。
* 
* @author Liu
* @date 2018年1月16日 下午9:45:37 
*  
*/
public class MultiThreadShareData {
	private int j = 0;
	
	public synchronized void inc(){
		j++;
		System.out.println(Thread.currentThread().getName() + "中 j = " + j);
	}
	
	public synchronized void dec(){
		j--;
		System.out.println(Thread.currentThread().getName() + "中 j = " + j);
	}
	
	/***
	* @Title: main 
	* @Description: TODO
	* @param @param args
	* @return void
	* @throws 
	*/
	public static void main(String[] args) {
		MultiThreadShareData multiThreadShareData = new MultiThreadShareData();
		for(int i = 0; i< 2; i++){
			new Thread(multiThreadShareData.new IncBusiness()).start();
			new Thread(multiThreadShareData.new DecBusiness()).start();
		}
	}
	
	class IncBusiness implements Runnable{

		/* (非 Javadoc) 
		* <p>Title: run</p> 
		* <p>Description: </p>  
		* @see java.lang.Runnable#run() 
		*/
		@Override
		public void run() {
			for(int i = 0; i< 5; i++){
				inc();
			}
		}
		
	}
	
	class DecBusiness implements Runnable{
		
		/* (非 Javadoc) 
		 * <p>Title: run</p> 
		 * <p>Description: </p>  
		 * @see java.lang.Runnable#run() 
		 */
		@Override
		public void run() {
			for(int i = 0; i< 5; i++){
				dec();
			}
		}
		
	}

}

        ③ 上面兩種方式的組合:將共享數據封裝在另一個對象中,每一個線程對共享數據的操做方法也分配到那個對象身上去完成,對象做爲這個外部類中的成員變量或方法中的局部變量,每一個線程的Runnable對象做爲外部類中的成員內部類或局部內部類。線程

        ④ 總之,要同步互斥的幾段代碼最好是分別放在幾個獨立的方法中,這些方法再放在同一個類中,這樣比較容易實現它們之間的同步互斥和通訊。設計

/** 
* @Title: MultiThreadShareData.java 
* @Package com.lh.threadtest.t7 
* @Description: TODO
* @author Liu 
* @date 2018年1月16日 下午9:45:37 
* @version V1.0 
*/
package com.lh.threadtest.t7;

/** 
* @ClassName: MultiThreadShareData 
* @Description: 多個線程之間共享數據的方式探討
* 
* 題目:
* 	設計4個線程,其中兩個線程每次對j增長1,另外兩個線程對j每次都減小1,寫出程序。
* 
* @author Liu
* @date 2018年1月16日 下午9:45:37 
*  
*/
public class MultiThreadShareData3 {
	
	/***
	* @Title: main 
	* @Description: TODO
	* @param @param args
	* @return void
	* @throws 
	*/
	public static void main(String[] args) {
		BusinessHandler business = new BusinessHandler();
		
		for(int i = 0; i< 2; i++){
			new Thread(new Runnable() {
				
				@Override
				public void run() {
					business.inc();
				}
			}).start();
			new Thread(new Runnable() {
				
				@Override
				public void run() {
					business.dec();
				}
			}).start();
		}
	}
}

class BusinessHandler{
	private int j = 0;
	
	public synchronized void inc(){
		j++;
		System.out.println(Thread.currentThread().getName() + "中 j = " + j);
	}
	
	public synchronized void dec(){
		j--;
		System.out.println(Thread.currentThread().getName() + "中 j = " + j);
	}
}

    三、極端且簡單的方式,即在任意一個類中定義一個static的變量,這將被全部線程共享。code

相關文章
相關標籤/搜索