日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

詳解Java Callable接口實現多線程的方式

瀏覽:78日期:2022-09-02 15:09:45

在Java 1.5以前,創建線程的2種方式,一種是直接繼承Thread,另外一種就是實現Runnable接口。無論我們以怎樣的形式實現多線程,都需要調用Thread類中的start方法去向操作系統請求io,cup等資源。因為線程run方法沒有返回值,如果需要獲取執行結果,就必須通過共享變量或者使用線程通信的方式來達到效果,這樣使用起來就比較麻煩。

而自從Java 1.5開始,就提供了Callable和Future,通過它們可以在任務執行完畢之后得到任務執行結果。

Callable和Future介紹

Callable接口代表一段可以調用并返回結果的代碼;Future接口表示異步任務,是還沒有完成的任務給出的未來結果。所以說Callable用于產生結果,Future用于獲取結果。

Callable接口使用泛型去定義它的返回類型。Executors類提供了一些有用的方法在線程池中執行Callable內的任務。由于Callable任務是并行的(并行就是整體看上去是并行的,其實在某個時間點只有一個線程在執行),我們必須等待它返回的結果。java.util.concurrent.Future對象為我們解決了這個問題。在線程池提交Callable任務后返回了一個Future對象,使用它可以知道Callable任務的狀態和得到Callable返回的執行結果。Future提供了get()方法讓我們可以等待Callable結束并獲取它的執行結果。

Callable與Runnable

java.lang.Runnable吧,它是一個接口,在它里面只聲明了一個run()方法:

public interface Runnable { public abstract void run();}

由于run()方法返回值為void類型,所以在執行完任務之后無法返回任何結果。Callable位于java.util.concurrent包下,它也是一個接口,在它里面也只聲明了一個方法,只不過這個方法叫做call():

public interface Callable<V> { /** * Computes a result, or throws an exception if unable to do so. * @return computed result * @throws Exception if unable to compute a result */ V call() throws Exception;}

可以看到,這是一個泛型接口,call()函數返回的類型就是傳遞進來的V類型。

那么怎么使用Callable呢?

一般情況下是配合ExecutorService來使用的,在ExecutorService接口中聲明了若干個submit方法的重載版本:

<T> Future<T> submit(Callable<T> task);<T> Future<T> submit(Runnable task, T result);Future<?> submit(Runnable task);

Future

Future就是對于具體的Runnable或者Callable任務的執行結果進行取消、查詢是否完成、獲取結果。必要時可以通過get方法獲取執行結果,該方法會阻塞直到任務返回結果。

Future類位于java.util.concurrent包下,它是一個接口

:<T> Future<T> submit(Callable<T> task);<T> Future<T> submit(Runnable task, T result);Future<?> submit(Runnable task);

在Future接口中聲明了5個方法,下面依次解釋每個方法的作用:

cancel方法 用來取消任務,如果取消任務成功則返回true,如果取消任務失敗則返回false。參數mayInterruptIfRunning表示是否允許取消正在執行卻沒有執行完畢的任務,如果設置true,則表示可以取消正在執行過程中的任務。如果任務已經完成,則無論mayInterruptIfRunning為true還是false,此方法肯定返回false,即如果取消已經完成的任務會返回false;如果任務正在執行,若mayInterruptIfRunning設置為true,則返回true,若mayInterruptIfRunning設置為false,則返回false;如果任務還沒有執行,則無論mayInterruptIfRunning為true還是false,肯定返回true。

isCancelled方法 表示任務是否被取消成功,如果在任務正常完成前被取消成功,則返回 true。

isDone方法 表示任務是否已經完成,若任務完成,則返回true;

get()方法 用來獲取執行結果,這個方法會產生阻塞,會一直等到任務執行完畢才返回;

get(long timeout, TimeUnit unit) 用來獲取執行結果,如果在指定時間內,還沒獲取到結果,就直接返回null。

也就是說Future提供了三種功能: 

1)判斷任務是否完成;

2)能夠中斷任務;

3)能夠獲取任務執行結果。

Future用于表示異步計算的結果。它的實現類有java.util.concurrent.FutureTask<V>和 javax.swing.SwingWorker<T,V>,Android平臺上如果不想分支線程阻塞主線程,又想取得分支線程的執行結果,可以用FutureTask。

FutureTask

Java的類是單繼承的設計,如果采用繼承Thread的方式實現多線程,則不能繼承其他的類,采用接口能夠更好的實現數據共享

FutureTask實現了RunnableFuture接口,這個接口的定義如下:

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

可以看到這個接口實現了Runnable和Future接口,接口中的具體實現由FutureTask來實現。這個類的兩個構造方法如下 :

public FutureTask(Callable<V> callable) { if (callable == null) throw new NullPointerException(); sync = new Sync(callable); } public FutureTask(Runnable runnable, V result) { sync = new Sync(Executors.callable(runnable, result)); }

如上提供了兩個構造函數,一個以Callable為參數,另外一個以Runnable為參數。這些類之間的關聯對于任務建模的辦法非常靈活,允許你基于FutureTask的Runnable特性(因為它實現了Runnable接口),把任務寫成Callable,然后封裝進一個由執行者調度并在必要時可以取消的FutureTask。

FutureTask可以由執行者調度,這一點很關鍵。它對外提供的方法基本上就是Future和Runnable接口的組合:get()、cancel、isDone()、isCancelled()和run(),而run()方法通常都是由執行者調用,我們基本上不需要直接調用它。FutureTask類同時又實現了Runnable接口,所以可以直接提交給Thread、Executors執行:

public class CallableAndFuture { public static void main(String[] args) { Callable<Integer> callable = new Callable<Integer>() { public Integer call() throws Exception { return new Random().nextInt(100); } }; FutureTask<Integer> future = new FutureTask<Integer>(callable); new Thread(future).start(); try { Thread.sleep(5000);// 可能做一些事情int result = future.get()); } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } } }

public class CallableAndFuture { public static void main(String[] args) { //ExecutorService.submit() ExecutorService threadPool = Executors.newSingleThreadExecutor(); Future<Integer> future = threadPool.submit(new Callable<Integer>() { public Integer call() throws Exception { return new Random().nextInt(100); } }); try { Thread.sleep(5000);// 可能做一些事情int result = future.get()); //Future.get() } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } }

如果要執行多個帶返回值的任務,并取得多個返回值,可用CompletionService:

CompletionService相當于Executor加上BlockingQueue,使用場景為當子線程并發了一系列的任務以后,主線程需要實時地取回子線程任務的返回值并同時順序地處理這些返回值,誰先返回就先處理誰。

public class CallableAndFuture { public static void main(String[] args) { ExecutorService threadPool = Executors.newCachedThreadPool(); CompletionService<Integer> cs = new ExecutorCompletionService<Integer>(threadPool); for(int i = 1; i < 5; i++) { final int taskID = i; //CompletionService.submit() cs.submit(new Callable<Integer>() { public Integer call() throws Exception { return taskID; } }); } // 可能做一些事情 for(int i = 1; i < 5; i++) { try { int result = cs.take().get()); //CompletionService.take()返回Future } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } } } }

或者不使用CompletionService:先創建一個裝Future類型的集合,用Executor提交的任務返回值添加到集合中,最后便利集合取出數據。

區別:

Future集合方法,submit的task不一定是按照加入自己維護的list順序完成的。從list中遍歷的每個Future對象并不一定處于完成狀態,這時調用get()方法就會被阻塞住,如果系統是設計成每個線程完成后就能根據其結果繼續做后面的事,這樣對于處于list后面的但是先完成的線程就會增加了額外的等待時間。

而CompletionService的實現是維護一個保存Future對象的BlockingQueue。只有當這個Future對象狀態是結束的時候,才會加入到這個Queue中,take()方法其實就是Producer-Consumer中的Consumer。它會從Queue中取出Future對象,如果Queue是空的,就會阻塞在那里,直到有完成的Future對象加入到Queue中。

所以,先完成的必定先被取出。這樣就減少了不必要的等待時間。

到此這篇關于詳解Java Callable接口實現多線程的方式的文章就介紹到這了,更多相關Java Callable接口多線程內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产日韩一区二区三免费高清| 色88888久久久久久影院| 成人日韩av| 欧美国产专区| 精品亚洲成人| 成人高清一区| 中文在线а√在线8| 国产不卡人人| 99久久九九| 亚洲一区二区毛片| 国产视频一区欧美| 蜜臀av一区二区在线免费观看 | 欧美综合精品| 日韩高清不卡一区二区| 69精品国产久热在线观看| 国产欧美自拍| 国产成人精品一区二区三区免费| 91日韩在线| 久久中文字幕二区| 午夜国产精品视频| 日韩一区精品视频| 欧美一区免费| 国产一区日韩| 成人精品国产亚洲| 亚洲二区在线| 老色鬼久久亚洲一区二区| 亚洲人成精品久久久| 国产麻豆一区| 日韩在线看片| 日韩中文字幕不卡| 亚洲最新av| 日本а中文在线天堂| 日韩不卡手机在线v区| 国产麻豆精品久久| 久久精品国产99| 成人日韩在线观看| 99国产精品久久久久久久| 日本久久二区| 国产亚洲精品美女久久| 波多视频一区| 综合激情一区| av资源新版天堂在线| 每日更新成人在线视频| 国产精品综合| 国产成人精品亚洲日本在线观看| 中国女人久久久| 国产精品亚洲欧美一级在线 | 日韩一区二区免费看| 欧美日韩国产一区二区在线观看| 欧美国产极品| 午夜国产一区二区| 欧美日韩一区自拍| 久久亚洲人体| 最新亚洲激情| 日韩精品视频网站| 综合一区在线| 国产伦久视频在线观看| 99久久亚洲精品| 成人国产精品一区二区网站| 天堂中文av在线资源库| 日韩高清一区在线| 午夜av一区| 欧美视频精品全部免费观看| 久久亚州av| 久久免费国产| 日韩影片在线观看| 国产精品亚洲综合久久| 中文字幕免费精品| 国产精品激情电影| 欧美 日韩 国产精品免费观看| 日韩欧美中文字幕一区二区三区| 日韩av资源网| 成人亚洲一区| 另类小说一区二区三区| 久久久久黄色| 日本不卡在线视频| 一区久久精品| 亚洲精品免费观看| 欧美三级精品| 日韩成人午夜精品| 亚洲高清二区| 美女国产精品久久久| 久久精品播放| 久久久国产精品网站| 亚洲欧美在线专区| 激情偷拍久久| 国产精品亚洲片在线播放| 蜜臀久久久99精品久久久久久| 久久精品电影| 高清一区二区| 国产福利一区二区精品秒拍| 免费美女久久99| 91精品一区二区三区综合在线爱 | 西西人体一区二区| 91亚洲国产高清| 青青草视频一区| 蜜臀av一区二区在线免费观看| 午夜精品久久久久久久久久蜜桃| 国产精品传媒麻豆hd| 亚欧成人精品| 亚洲夜间福利| 涩涩av在线| 91麻豆国产自产在线观看亚洲| 麻豆精品久久| 国产精品天天看天天狠| 欧美一区二区三区久久| 亚洲精品无播放器在线播放| 奶水喷射视频一区| 午夜国产精品视频免费体验区| 成人午夜亚洲| 老司机精品视频在线播放| 88久久精品| 蜜桃av一区二区| 性色一区二区| 中文欧美日韩| 欧美日韩亚洲一区二区三区在线| 亚洲午夜黄色| 成人一二三区| 麻豆网站免费在线观看| 国产一区清纯| 91久久在线| 亚洲二区在线| 激情黄产视频在线免费观看| 日本午夜精品| 亚洲一区二区三区在线免费| 中文字幕成在线观看| 黄色精品视频| 午夜在线视频观看日韩17c| 国产成年精品| 亚洲欧美日韩精品一区二区 | 欧美一级二级视频| 久久国产精品久久久久久电车 | 日韩国产欧美在线播放| 香蕉精品久久| 欧美日韩一区二区高清| 国产成人精选| 日韩一区二区三区四区五区| 国产日韩高清一区二区三区在线| 日韩精品免费一区二区夜夜嗨| 桃色一区二区| 国产美女亚洲精品7777| 91偷拍一区二区三区精品| 亚州av乱码久久精品蜜桃| 欧美国产精品| 亚洲a成人v| 开心激情综合| 精品在线91| 福利在线一区| 久久久精品网| 在线视频免费在线观看一区二区| 亚洲色诱最新| 日本午夜精品一区二区三区电影 | 女主播福利一区| 黄色成人在线网址| 蜜臀久久久99精品久久久久久| 久久久久国产精品一区三寸| 综合日韩av| 久久不射网站| 视频一区视频二区中文| 宅男在线一区| 精品久久免费| 激情欧美一区| 欧美一区久久| 国产精品不卡| 先锋影音国产一区| 成人国产精选| 国产精品普通话对白| 国产精品黄色| 亚洲欧美视频| 亚洲国产福利| 色一区二区三区四区| 亚洲资源在线| 中文字幕亚洲精品乱码| 女主播福利一区| 国产高清久久| 久久精品国语| 高潮久久久久久久久久久久久久| 欧美另类中文字幕| 国产欧美日韩免费观看| 日韩国产91| 美女视频黄免费的久久| 国产精品一区二区三区www| 首页国产精品| 亚洲免费播放| 日韩有码av| 中文字幕在线免费观看视频| 亚洲免费在线| 久久99久久久精品欧美| 免费欧美一区| 欧美久久久网站| 亲子伦视频一区二区三区| 亚洲字幕久久| 日韩伦理在线一区| 亚州精品视频| 久久高清免费| 国产亚洲久久| 日韩网站在线| 麻豆中文一区二区| 激情五月综合| 久久精品国内一区二区三区|