欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java多線程編程同步器Future和FutureTask解析及代碼示例

 更新時間:2017年11月20日 14:24:12   作者:zheng12tian  
這篇文章主要介紹了java多線程編程同步器Future和FutureTask解析及代碼示例,對二者進行了詳細介紹,分析了future的源碼,最后展示了相關實例代碼,具有一定參考價值 ,需要的朋友可以了解下。

publicinterfaceFuture<V>Future表示異步計算的結果。它提供了檢查計算是否完成的方法,以等待計算的完成,并獲取計算的結果。計算完成后只能使用get方法來獲取結果,如有必要,計算完成前可以阻塞此方法。取消則由cancel方法來執(zhí)行。還提供了其他方法,以確定任務是正常完成還是被取消了。一旦計算完成,就不能再取消計算。如果為了可取消性而使用Future但又不提供可用的結果,則可以聲明Future<?>形式類型、并返回null作為底層任務的結果。

Future主要定義了5個方法:

1)booleancancel(booleanmayInterruptIfRunning):試圖取消對此任務的執(zhí)行。如果任務已完成、或已取消,或者由于某些其他原因而無法取消,則此嘗試將失敗。當調用cancel時,如果調用成功,而此任務尚未啟動,則此任務將永不運行。如果任務已經(jīng)啟動,則mayInterruptIfRunning參數(shù)確定是否應該以試圖停止任務的方式來中斷執(zhí)行此任務的線程。此方法返回后,對isDone()的后續(xù)調用將始終返回true。如果此方法返回true,則對isCancelled()的后續(xù)調用將始終返回true。

2)booleanisCancelled():如果在任務正常完成前將其取消,則返回true。

3)booleanisDone():如果任務已完成,則返回true??赡苡捎谡=K止、異常或取消而完成,在所有這些情況中,此方法都將返回true。

4)Vget()throwsInterruptedException,ExecutionException:如有必要,等待計算完成,然后獲取其結果。

5)Vget(longtimeout,TimeUnitunit)throwsInterruptedException,ExecutionException,TimeoutException:如有必要,最多等待為使計算完成所給定的時間之后,獲取其結果(如果結果可用)。

Future<V>接口

Future<V>接口是用來獲取異步計算結果的,說白了就是對具體的Runnable或者Callable對象任務執(zhí)行的結果進行獲取(get()),取消(cancel()),判斷是否完成等操作。我們看看Future接口的源碼:

public interface Future<V> { 
 boolean cancel(boolean mayInterruptIfRunning); 
 boolean isCancelled(); 
 boolean isDone(); 
 V get() throws InterruptedException, ExecutionException; 
 V get(long timeout, TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException; 
} 

方法解析:

Vget():獲取異步執(zhí)行的結果,如果沒有結果可用,此方法會阻塞直到異步計算完成。

Vget(Longtimeout,TimeUnitunit):獲取異步執(zhí)行結果,如果沒有結果可用,此方法會阻塞,但是會有時間限制,如果阻塞時間超過設定的timeout時間,該方法將拋出異常。

booleanisDone():如果任務執(zhí)行結束,無論是正常結束或是中途取消還是發(fā)生異常,都返回true。

booleanisCanceller():如果任務完成前被取消,則返回true。

booleancancel(booleanmayInterruptRunning):如果任務還沒開始,執(zhí)行cancel(...)方法將返回false;如果任務已經(jīng)啟動,執(zhí)行cancel(true)方法將以中斷執(zhí)行此任務線程的方式來試圖停止任務,如果停止成功,返回true;當任務已經(jīng)啟動,執(zhí)行cancel(false)方法將不會對正在執(zhí)行的任務線程產生影響(讓線程正常執(zhí)行到完成),此時返回false;當任務已經(jīng)完成,執(zhí)行cancel(...)方法將返回false。mayInterruptRunning參數(shù)表示是否中斷執(zhí)行中的線程。

通過方法分析我們也知道實際上Future提供了3種功能:(1)能夠中斷執(zhí)行中的任務(2)判斷任務是否執(zhí)行完成(3)獲取任務執(zhí)行完成后額結果。

但是我們必須明白Future只是一個接口,我們無法直接創(chuàng)建對象,因此就需要其實現(xiàn)類FutureTask登場啦。

FutureTask類

FutureTask可用于異步獲取執(zhí)行結果或取消執(zhí)行任務的場景。通過傳入Runnable或者Callable的任務給FutureTask,直接調用其run方法或者放入線程池執(zhí)行,之后可以在外部通過FutureTask的get方法異步獲取執(zhí)行結果,因此,F(xiàn)utureTask非常適合用于耗時的計算,主線程可以在完成自己的任務后,再去獲取結果。另外,F(xiàn)utureTask還可以確保即使調用了多次run方法,它都只會執(zhí)行一次Runnable或者Callable任務,或者通過cancel取消FutureTask的執(zhí)行等。

我們來看看FutureTask的實現(xiàn)

public class FutureTask<V> implements RunnableFuture<V> { 

FutureTask類實現(xiàn)了RunnableFuture接口,我們看一下RunnableFuture接口的實現(xiàn):

public interface RunnableFuture<V> extends Runnable, Future<V> { 
 void run(); 
} 

分析:FutureTask除了實現(xiàn)了Future接口外還實現(xiàn)了Runnable接口,因此FutureTask也可以直接提交給Executor執(zhí)行。當然也可以調用線程直接執(zhí)行(FutureTask.run())。接下來我們根據(jù)FutureTask.run()的執(zhí)行時機來分析其所處的3種狀態(tài):

(1)未啟動,F(xiàn)utureTask.run()方法還沒有被執(zhí)行之前,F(xiàn)utureTask處于未啟動狀態(tài),當創(chuàng)建一個FutureTask,而且沒有執(zhí)行FutureTask.run()方法前,這個FutureTask也處于未啟動狀態(tài)。

(2)已啟動,F(xiàn)utureTask.run()被執(zhí)行的過程中,F(xiàn)utureTask處于已啟動狀態(tài)。

(3)已完成,F(xiàn)utureTask.run()方法執(zhí)行完正常結束,或者被取消或者拋出異常而結束,F(xiàn)utureTask都處于完成狀態(tài)。

FutureTask類是Future的一個實現(xiàn),并實現(xiàn)了Runnable,所以可通過Excutor(線程池)來執(zhí)行。也可傳遞給Thread對象執(zhí)行。如果在主線程中需要執(zhí)行比較耗時的操作時,但又不想阻塞主線程時,可以把這些作業(yè)交給Future對象在后臺完成,當主線程將來需要時,就可以通過Future對象獲得后臺作業(yè)的計算結果或者執(zhí)行狀態(tài)。下面的例子模擬一個會計算賬的過程,主線程已經(jīng)獲得其他帳戶的總額了,為了不讓主線程等待PrivateAccount類的計算結果的返回而啟用新的線程去處理,并使用FutureTask對象來監(jiān)控,這樣,主線程還可以繼續(xù)做其他事情,最后需要計算總額的時候再嘗試去獲得privateAccount的信息。

package test;
import java.util.Random;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
/**
 *
 * @author Administrator
 *
 */
@SuppressWarnings("all")
public class FutureTaskDemo {
	public static void main(String[] args) {
		// 初始化一個Callable對象和FutureTask對象
		Callable pAccount = new PrivateAccount();
		FutureTask futureTask = new FutureTask(pAccount);
		// 使用futureTask創(chuàng)建一個線程
		Thread pAccountThread = new Thread(futureTask);
		System.out.println("futureTask線程現(xiàn)在開始啟動,啟動時間為:" + System.nanoTime());
		pAccountThread.start();
		System.out.println("主線程開始執(zhí)行其他任務");
		// 從其他賬戶獲取總金額
		int totalMoney = new Random().nextint(100000);
		System.out.println("現(xiàn)在你在其他賬戶中的總金額為" + totalMoney);
		System.out.println("等待私有賬戶總金額統(tǒng)計完畢...");
		// 測試后臺的計算線程是否完成,如果未完成則等待
		while (!futureTask.isDone()) {
			try {
				Thread.sleep(500);
				System.out.println("私有賬戶計算未完成繼續(xù)等待...");
			}
			catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
		System.out.println("futureTask線程計算完畢,此時時間為" + System.nanoTime());
		Integer privateAccountMoney = null;
		try {
			privateAccountMoney = (Integer) futureTask.get();
		}
		catch (InterruptedException e) {
			e.printStackTrace();
		}
		catch (ExecutionException e) {
			e.printStackTrace();
		}
		System.out.println("您現(xiàn)在的總金額為:" + totalMoney + privateAccountMoney.intValue());
	}
}
@SuppressWarnings("all")
class PrivateAccount implements Callable {
	Integer totalMoney;
	@Override
		public Object call() throws Exception {
		Thread.sleep(5000);
		totalMoney = new Integer(new Random().nextint(10000));
		System.out.println("您當前有" + totalMoney + "在您的私有賬戶中");
		return totalMoney;
	}
}

運行結果 

futureTask線程現(xiàn)在開始啟動,啟動時間為:3098040622063 
主線程開始執(zhí)行其他任務 
現(xiàn)在你在其他賬戶中的總金額為56983 
等待私有賬戶總金額統(tǒng)計完畢... 
私有賬戶計算未完成繼續(xù)等待... 
私有賬戶計算未完成繼續(xù)等待... 
私有賬戶計算未完成繼續(xù)等待... 
私有賬戶計算未完成繼續(xù)等待... 
私有賬戶計算未完成繼續(xù)等待... 
私有賬戶計算未完成繼續(xù)等待... 
私有賬戶計算未完成繼續(xù)等待... 
私有賬戶計算未完成繼續(xù)等待... 
私有賬戶計算未完成繼續(xù)等待... 
您當前有3345在您的私有賬戶中 
私有賬戶計算未完成繼續(xù)等待... 
futureTask線程計算完畢,此時時間為3103072404138 
您現(xiàn)在的總金額為:569833345

總結

以上就是本文關于java多線程編程同步器Future和FutureTask解析及代碼示例的全部內容,希望對大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站:

Java多線程之顯示鎖和內置鎖總結詳解

Java多線程中斷機制三種方法及示例

java多線程編程實例

如有不足之處,歡迎留言指出。

相關文章

  • 詳解Java中l(wèi)og4j.properties配置與加載應用

    詳解Java中l(wèi)og4j.properties配置與加載應用

    這篇文章主要介紹了 log4j.properties配置與加載應用的相關資料,需要的朋友可以參考下
    2018-02-02
  • Java中的 VO,BO,DO 對象命名問題小結

    Java中的 VO,BO,DO 對象命名問題小結

    本文講解VO,BO,DO 的作用以及如何使用,分析了如何消除三者之間重復的代碼,同樣結合現(xiàn)實生活中領導配秘書來類比講解,對Java  VO  對象命名相關知識感興趣的朋友一起看看吧
    2024-01-01
  • 輕松搞定SpringBoot JPA使用配置過程詳解

    輕松搞定SpringBoot JPA使用配置過程詳解

    Spring Boot是由Pivotal團隊提供的全新框架,該框架使用了特定的方式來進行配置,它默認配置了很多框架的使用方式,就像 Maven整合了所有的Jar包,Spring Boot 整合了所有的框架
    2021-06-06
  • Java 反轉帶頭結點的單鏈表并顯示輸出的實現(xiàn)過程

    Java 反轉帶頭結點的單鏈表并顯示輸出的實現(xiàn)過程

    這篇文章主要介紹了Java 反轉帶頭結點的單鏈表并顯示輸出,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-11-11
  • spring boot集成mongodb的增刪改查的示例代碼

    spring boot集成mongodb的增刪改查的示例代碼

    這篇文章主要介紹了spring boot集成mongodb的增刪改查的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-03-03
  • Java Spring動態(tài)生成Mysql存儲過程詳解

    Java Spring動態(tài)生成Mysql存儲過程詳解

    這篇文章主要介紹了Java Spring動態(tài)生成Mysql存儲過程詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-06-06
  • springboot解決java.lang.ArrayStoreException異常

    springboot解決java.lang.ArrayStoreException異常

    這篇文章介紹了springboot解決java.lang.ArrayStoreException異常的方法,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-12-12
  • IO流:java中解碼和編碼出現(xiàn)亂碼說明以及代碼實現(xiàn)方法

    IO流:java中解碼和編碼出現(xiàn)亂碼說明以及代碼實現(xiàn)方法

    最近使用Java編寫一些讀取文件的小工具的時候,經(jīng)常與IO流打交道,但是自己對IO流的理解不是特別深刻,趁機整理下,這篇文章主要給大家介紹了關于IO流:java中解碼和編碼出現(xiàn)亂碼說明以及代碼實現(xiàn)的相關資料,需要的朋友可以參考下
    2023-11-11
  • Java 坐標系相互轉換方式

    Java 坐標系相互轉換方式

    這篇文章主要介紹了Java中的坐標系相互轉換方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • 經(jīng)典再現(xiàn) 基于JAVA平臺開發(fā)坦克大戰(zhàn)游戲

    經(jīng)典再現(xiàn) 基于JAVA平臺開發(fā)坦克大戰(zhàn)游戲

    經(jīng)典再現(xiàn),這篇文章主要介紹了基于JAVA平臺開發(fā)坦克大戰(zhàn)游戲的相關代碼,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2016-06-06

最新評論