亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

Java線程池

2019-11-14 20:49:09
字體:
來源:轉載
供稿:網友
java線程池

線程池的作用

線程池作用就是限制系統中執行線程的數量。

根據系統的環境情況,可以自動或手動設置線程數量,達到運行的最佳效果;少了浪費了系統資源,多了造成系統擁擠效率不高。用線程池控制線程數量,其他線程排隊等候。一個任務執行完畢,再從隊列的中取最前面的任務開始執行。若隊列中沒有等待進程,線程池的這一資源處于等待。當一個新任務需要運行時,如果線程池中有等待的工作線程,就可以開始運行了;否則進入等待隊列。

為什么要用線程池

1.減少了創建和銷毀線程的次數,每個工作線程都可以被重復利用,可執行多個任務。

2.可以根據系統的承受能力,調整線程池中工作線線程的數目,防止因為消耗過多的內存,而把服務器累趴下(每個線程需要大約1MB內存,線程開的越多,消耗的內存也就越大,最后死機)。

Java里面線程池的頂級接口是Executor,但是嚴格意義上講Executor并不是一個線程池,而只是一個執行線程的工具。真正的線程池接口是ExecutorService。

比較重要的幾個類:

ExecutorService

真正的線程池接口。

ScheduledExecutorService

能和Timer/TimerTask類似,解決那些需要任務重復執行的問題。

ThreadPoolExecutor

ExecutorService的默認實現。

ScheduledThreadPoolExecutor

繼承ThreadPoolExecutor的ScheduledExecutorService接口實現,周期性任務調度的類實現。

要配置一個線程池是比較復雜的,尤其是對于線程池的原理不是很清楚的情況下,很有可能配置的線程池不是較優的,因此在Executors類里面提供了一些靜態工廠,生成一些常用的線程池。

1. newSingleThreadExecutor

創建一個單線程的線程池。這個線程池只有一個線程在工作,也就是相當于單線程串行執行所有任務。如果這個唯一的線程因為異常結束,那么會有一個新的線程來替代它。此線程池保證所有任務的執行順序按照任務的提交順序執行。

2.newFixedThreadPool

創建固定大小的線程池。每次提交一個任務就創建一個線程,直到線程達到線程池的最大大小。線程池的大小一旦達到最大值就會保持不變,如果某個線程因為執行異常而結束,那么線程池會補充一個新線程。

3. newCachedThreadPool

創建一個可緩存的線程池。如果線程池的大小超過了處理任務所需要的線程,

那么就會回收部分空閑(60秒不執行任務)的線程,當任務數增加時,此線程池又可以智能的添加新線程來處理任務。此線程池不會對線程池大小做限制,線程池大小完全依賴于操作系統(或者說JVM)能夠創建的最大線程大小。

4.newScheduledThreadPool

創建一個大小無限的線程池。此線程池支持定時以及周期性執行任務的需求。

newSingleThreadExecutor

MyThread.java

public class MyThread extends Thread {    @Override    publicvoid run() {        System.out.PRintln(Thread.currentThread().getName() + "正在執行。。。");    }}

TestSingleThreadExecutor.java

public class TestSingleThreadExecutor {    public static void main(String[] args) {        //創建一個可重用固定線程數的線程池        ExecutorService pool = Executors. newSingleThreadExecutor();        //創建實現了Runnable接口對象,Thread對象當然也實現了Runnable接口        Thread t1 = new MyThread();        Thread t2 = new MyThread();        Thread t3 = new MyThread();        Thread t4 = new MyThread();        Thread t5 = new MyThread();        //將線程放入池中進行執行        pool.execute(t1);        pool.execute(t2);        pool.execute(t3);        pool.execute(t4);        pool.execute(t5);        //關閉線程池        pool.shutdown();    }}

輸出結果

pool-1-thread-1正在執行。。。pool-1-thread-1正在執行。。。pool-1-thread-1正在執行。。。pool-1-thread-1正在執行。。。pool-1-thread-1正在執行。。。

newFixedThreadPool

TestFixedThreadPool.Java

public class TestFixedThreadPool {    public static void main(String[] args) {        //創建一個可重用固定線程數的線程池        ExecutorService pool = Executors.newFixedThreadPool(2);        //創建實現了Runnable接口對象,Thread對象當然也實現了Runnable接口        Thread t1 = new MyThread();        Thread t2 = new MyThread();        Thread t3 = new MyThread();        Thread t4 = new MyThread();        Thread t5 = new MyThread();        //將線程放入池中進行執行         pool.execute(t1);        pool.execute(t2);        pool.execute(t3);        pool.execute(t4);        pool.execute(t5);        //關閉線程池        pool.shutdown();    }}

輸出結果

pool-1-thread-1正在執行。。。pool-1-thread-2正在執行。。。pool-1-thread-1正在執行。。。pool-1-thread-2正在執行。。。pool-1-thread-1正在執行。。。

newCachedThreadPool

TestCachedThreadPool.java

public class TestCachedThreadPool {    public static void main(String[] args) {        //創建一個可重用固定線程數的線程池        ExecutorService pool = Executors.newCachedThreadPool();        //創建實現了Runnable接口對象,Thread對象當然也實現了Runnable接口        Thread t1 = new MyThread();        Thread t2 = new MyThread();        Thread t3 = new MyThread();        Thread t4 = new MyThread();        Thread t5 = new MyThread();        //將線程放入池中進行執行        pool.execute(t1);        pool.execute(t2);        pool.execute(t3);        pool.execute(t4);        pool.execute(t5);        //關閉線程池        pool.shutdown();    }}

輸出結果:

pool-1-thread-2正在執行。。。pool-1-thread-4正在執行。。。pool-1-thread-3正在執行。。。pool-1-thread-1正在執行。。。pool-1-thread-5正在執行。。。

newScheduledThreadPool

TestScheduledThreadPoolExecutor.java

public class TestScheduledThreadPoolExecutor {    public static void main(String[] args) {        ScheduledThreadPoolExecutor exec = new ScheduledThreadPoolExecutor(1);        exec.scheduleAtFixedRate(new Runnable() {//每隔一段時間就觸發異常                      @Override                      publicvoid run() {                           //throw new RuntimeException();                           System.out.println("================");                      }                  }, 1000, 5000, TimeUnit.MILLISECONDS);        exec.scheduleAtFixedRate(new Runnable() {//每隔一段時間打印系統時間,證明兩者是互不影響的                      @Override                      publicvoid run() {                           System.out.println(System.nanoTime());                      }                  }, 1000, 2000, TimeUnit.MILLISECONDS);    }}

輸出結果

================838464454951683866438290348388643830710================839064385138383926438793198400643939383

ThreadPoolExecutor詳解

hreadPoolExecutor的完整構造方法的簽名是:

ThreadPoolExecutor(int corePoolSize, int maximumPoolSize, long keepAliveTime, TimeUnit unit, BlockingQueue<Runnable> workQueue, ThreadFactory threadFactory, RejectedExecutionHandler handler) .

corePoolSize - 池中所保存的線程數,包括空閑線程。

maximumPoolSize-池中允許的最大線程數。

keepAliveTime - 當線程數大于核心時,此為終止前多余的空閑線程等待新任務的最長時間。

unit - keepAliveTime 參數的時間單位。

workQueue - 執行前用于保持任務的隊列。此隊列僅保持由 execute方法提交的 Runnable任務。

threadFactory - 執行程序創建新線程時使用的工廠。

handler - 由于超出線程范圍和隊列容量而使執行被阻塞時所使用的處理程序。

ThreadPoolExecutor是Executors類的底層實現。

在JDK幫助文檔中,有如此一段話:

“強烈建議程序員使用較為方便的Executors工廠方法Executors.newCachedThreadPool()(無界線程池,可以進行自動線程回收)、Executors.newFixedThreadPool(int)(固定大小線程池)Executors.newSingleThreadExecutor()(單個后臺線程)

它們均為大多數使用場景預定義了設置。”

下面介紹一下幾個類的源碼

ExecutorService newFixedThreadPool (int nThreads):固定大小線程池。

可以看到,corePoolSize和maximumPoolSize的大小是一樣的(實際上,后面會介紹,如果使用無界queue的話maximumPoolSize參數是沒有意義的),keepAliveTime和unit的設值表名什么?-就是該實現不想keep alive!最后的BlockingQueue選擇了LinkedBlockingQueue,該queue有一個特點,他是無界的。

public static ExecutorService newFixedThreadPool(int nThreads) {                return new ThreadPoolExecutor(nThreads, nThreads,    0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<Runnable>());            }

ExecutorService newSingleThreadExecutor():單線程

public static ExecutorService newSingleThreadExecutor() {               return new FinalizableDelegatedExecutorService                  (new ThreadPoolExecutor(1, 1, 0L, TimeUnit.MILLISECONDS,                        new LinkedBlockingQueue<Runnable>()));           }

ExecutorService newCachedThreadPool():無界線程池,可以進行自動線程回收

這個實現就有意思了。首先是無界的線程池,所以我們可以發現maximumPoolSize為big big。其次BlockingQueue的選擇上使用SynchronousQueue。可能對于該BlockingQueue有些陌生,簡單說:該QUEUE中,每個插入操作必須等待另一個線程的對應移除操作。

public static ExecutorService newCachedThreadPool() {           return new ThreadPoolExecutor(0, Integer.MAX_VALUE,   60L, TimeUnit.SECONDS,                  new SynchronousQueue<Runnable>());       }

先從BlockingQueue<Runnable> workQueue這個入參開始說起。在JDK中,其實已經說得很清楚了,一共有三種類型的queue。

所有BlockingQueue 都可用于傳輸和保持提交的任務??梢允褂么岁犃信c池大小進行交互:

如果運行的線程少于 corePoolSize,則 Executor始終首選添加新的線程,而不進行排隊。(如果當前運行的線程小于corePoolSize,則任務根本不會存放,添加到queue中,而是直接抄家伙(thread)開始運行)

如果運行的線程等于或多于 corePoolSize,則 Executor始終首選將請求加入隊列,而不添加新的線程。

如果無法將請求加入隊列,則創建新的線程,除非創建此線程超出 maximumPoolSize,在這種情況下,任務將被拒絕。

queue上的三種類型

排隊有三種通用策略:

直接提交。

工作隊列的默認選項是 SynchronousQueue,它將任務直接提交給線程而不保持它們。在此,如果不存在可用于立即運行任務的線程,則試圖把任務加入隊列將失敗,因此會構造一個新的線程。此策略可以避免在處理可能具有內部依賴性的請求集時出現鎖。直接提交通常要求無界maximumPoolSizes 以避免拒絕新提交的任務。當命令以超過隊列所能處理的平均數連續到達時,此策略允許無界線程具有增長的可能性。

無界隊列。

使用無界隊列(例如,不具有預定義容量的 LinkedBlockingQueue)將導致在所有 corePoolSize 線程都忙時新任務在隊列中等待。這樣,創建的線程就不會超過 corePoolSize。(因此,maximumPoolSize的值也就無效了。)當每個任務完全獨立于其他任務,即任務執行互不影響時,適合于使用無界隊列;例如,在 Web頁服務器中。這種排隊可用于處理瞬態突發請求,當命令以超過隊列所能處理的平均數連續到達時,此策略允許無界線程具有增長的可能性。

有界隊列。

當使用有限的 maximumPoolSizes時,有界隊列(如 ArrayBlockingQueue)有助于防止資源耗盡,但是可能較難調整和控制。隊列大小和最大池大小可能需要相互折衷:使用大型隊列和小型池可以最大限度地降低 CPU 使用率、操作系統資源和上下文切換開銷,但是可能導致人工降低吞吐量。如果任務頻繁阻塞(例如,如果它們是 I/O邊界),則系統可能為超過您許可的更多線程安排時間。使用小型隊列通常要求較大的池大小,CPU使用率較高,但是可能遇到不可接受的調度開銷,這樣也會降低吞吐量。

BlockingQueue的選擇

  • 使用直接提交策略,也即SynchronousQueue

首先SynchronousQueue是無界的,也就是說他存數任務的能力是沒有限制的,但是由于該Queue本身的特性,在某次添加元素后必須等待其他線程取走后才能繼續添加。在這里不是核心線程便是新創建的線程,但是我們試想一樣下,下面的場景。

我們使用一下參數構造ThreadPoolExecutor:

new ThreadPoolExecutor(                    2, 3, 30, TimeUnit.SECONDS,                     new  SynchronousQueue<Runnable>(),                    new RecorderThreadFactory("CookieRecorderPool"),                new ThreadPoolExecutor.CallerRunsPolicy());

當核心線程已經有2個正在運行.

1、此時繼續來了一個任務(A),根據前面介紹的“如果運行的線程等于或多于 corePoolSize,則 Executor始終首選將請求加入隊列,而不添加新的線程。”,所以A被添加到queue中。

2、又來了一個任務(B),且核心2個線程還沒有忙完,OK,接下來首先嘗試1中描述,但是由于使用的SynchronousQueue,所以一定無法加入進去。

3、此時便滿足了上面提到的“如果無法將請求加入隊列,則創建新的線程,除非創建此線程超出maximumPoolSize,在這種情況下,任務將被拒絕。”,所以必然會新建一個線程來運行這個任務。

4、暫時還可以,但是如果這三個任務都還沒完成,連續來了兩個任務,第一個添加入queue中,后一個呢?queue中無法插入,而線程數達到了maximumPoolSize,所以只好執行異常策略了。

所以在使用SynchronousQueue通常要求maximumPoolSize是無界的,這樣就可以避免上述情況發生(如果希望限制就直接使用有界隊列)。對于使用SynchronousQueue的作用jdk中寫的很清楚:此策略可以避免在處理可能具有內部依賴性的請求集時出現鎖。

什么意思?如果你的任務A1,A2有內部關聯,A1需要先運行,那么先提交A1,再提交A2,當使用SynchronousQueue我們可以保證,A1必定先被執行,在A1么有被執行前,A2不可能添加入queue中。

  • 使用無界隊列策略,即LinkedBlockingQueue

這個是最為復雜的使用,所以JDK不推薦使用也有些道理。與上面的相比,最大的特點便是可以防止資源耗盡的情況發生。

舉例來說,請看如下構造方法:

new ThreadPoolExecutor(    2, 4, 30, TimeUnit.SECONDS,    new ArrayBlockingQueue<Runnable>(2),    new RecorderThreadFactory("CookieRecorderPool"),    new ThreadPoolExecutor.CallerRunsPolicy());

假設,所有的任務都永遠無法執行完。

對于首先來的A,B來說直接運行,接下來,如果來了C,D,他們會被放到queue中,如果接下來再來E,F,則增加線程運行E,F。但是如果再來任務,隊列無法再接受了,線程數也到達最大的限制了,所以就會使用拒絕策略來處理。

keepAliveTime

jdk中的解釋是:當線程數大于核心時,此為終止前多余的空閑線程等待新任務的最長時間。

有點拗口,其實這個不難理解,在使用了“池”的應用中,大多都有類似的參數需要配置。比如數據庫連接池,DBCP中的maxIdle,minIdle參數。

什么意思?接著上面的解釋,后來向老板派來的工人始終是“借來的”,俗話說“有借就有還”,但這里的問題就是什么時候還了,如果借來的工人剛完成一個任務就還回去,后來發現任務還有,那豈不是又要去借?這一來一往,老板肯定頭也大死了。

合理的策略:既然借了,那就多借一會兒。直到“某一段”時間后,發現再也用不到這些工人時,便可以還回去了。這里的某一段時間便是keepAliveTime的含義,TimeUnit為keepAliveTime值的度量。

RejectedExecutionHandler

另一種情況便是,即使向老板借了工人,但是任務還是繼續過來,還是忙不過來,這時整個隊伍只好拒絕接受了。

RejectedExecutionHandler接口提供了對于拒絕任務的處理的自定方法的機會。在ThreadPoolExecutor中已經默認包含了4中策略,因為源碼非常簡單,這里直接貼出來。

  • CallerRunsPolicy:線程調用運行該任務的 execute 本身。此策略提供簡單的反饋控制機制,能夠減緩新任務的提交速度。
public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {           if (!e.isShutdown()) {               r.run();           }       }

這個策略顯然不想放棄執行任務。但是由于池中已經沒有任何資源了,那么就直接使用調用該execute的線程本身來執行。

  • AbortPolicy:處理程序遭到拒絕將拋出運行時RejectedExecutionException
public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {           throw new RejectedExecutionException();       }

這種策略直接拋出異常,丟棄任務。

  • DiscardPolicy:不能執行的任務將被刪除
public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {       }

這種策略和AbortPolicy幾乎一樣,也是丟棄任務,只不過他不拋出異常。

  • DiscardOldestPolicy:如果執行程序尚未關閉,則位于工作隊列頭部的任務將被刪除,然后重試執行程序(如果再次失敗,則重復此過程)
public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {           if (!e.isShutdown()) {               e.getQueue().poll();               e.execute(r);           }       }

該策略就稍微復雜一些,在pool沒有關閉的前提下首先丟掉緩存在隊列中的最早的任務,然后重新嘗試運行該任務。這個策略需要適當小心。

設想:如果其他線程都還在運行,那么新來任務踢掉舊任務,緩存在queue中,再來一個任務又會踢掉queue中最老任務。

總結

keepAliveTime和maximumPoolSize及BlockingQueue的類型均有關系。如果BlockingQueue是無界的,那么永遠不會觸發maximumPoolSize,自然keepAliveTime也就沒有了意義。

反之,如果核心數較小,有界BlockingQueue數值又較小,同時keepAliveTime又設的很小,如果任務頻繁,那么系統就會頻繁的申請回收線程。

我是天王蓋地虎的分割線

參考:http://blog.csdn.net/sd0902/article/details/8395677


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲电影免费观看高清完整版在线观看| 亚洲少妇中文在线| 亚洲激情视频在线| 欧美精品一区二区三区国产精品| 色小说视频一区| 欧美精品一区二区免费| 国产精品wwwwww| 久久成人一区二区| 欧美国产日产韩国视频| 亚洲韩国日本中文字幕| 欧美成人一区在线| 亚洲欧美国产va在线影院| 久久精品国产久精国产思思| 国产精品免费一区二区三区都可以| 欧美一级淫片videoshd| 亚洲欧美一区二区精品久久久| 亚洲黄页网在线观看| 亚洲国产精品一区二区久| 亚洲www在线| 中文亚洲视频在线| 日韩av电影手机在线观看| 国产成人亚洲精品| 亚洲精品久久久一区二区三区| 国产网站欧美日韩免费精品在线观看| 亚洲第一区在线观看| 亚洲欧洲成视频免费观看| 国产精品激情av电影在线观看| 午夜精品蜜臀一区二区三区免费| 一区二区三区视频免费| 久久久久久久久久久成人| 日韩在线观看你懂的| 国产欧美精品日韩| 国产亚洲精品综合一区91| 欧美高清视频免费观看| 亚洲成人久久电影| 一区国产精品视频| 精品国产一区二区三区久久狼5月| 成人性生交大片免费看视频直播| 亚洲欧美中文另类| 久久在线观看视频| 日韩精品视频在线| 亚洲欧美国产va在线影院| 亚洲第一福利网| 国产精品久久久久久久久久新婚| 性夜试看影院91社区| 国产精品久久久久久久av大片| 亚洲成人精品视频在线观看| 欧美亚洲成人xxx| 亚洲欧美中文另类| 一区二区三区视频免费在线观看| 国产做受高潮69| 亚洲深夜福利视频| 亚洲成人黄色网址| 大伊人狠狠躁夜夜躁av一区| 欧美视频中文在线看| 久久韩剧网电视剧| 久久精品国亚洲| 国产成人免费av电影| 日韩欧美大尺度| 亚洲偷欧美偷国内偷| 97视频在线播放| 日韩国产中文字幕| 欧美性xxxx极品hd满灌| 成人网址在线观看| 中文字幕精品国产| 日韩激情视频在线播放| 国产成人a亚洲精品| 亚洲人成电影网站色…| 久久久精品一区二区| 国产日本欧美一区| 992tv成人免费视频| 久久网福利资源网站| 一区二区三区无码高清视频| 色妞欧美日韩在线| 一本一道久久a久久精品逆3p| 亚洲欧美日韩另类| 91国自产精品中文字幕亚洲| 亚洲最大av在线| 国产精品99免视看9| 亚洲精品视频免费在线观看| 国产男女猛烈无遮挡91| 日韩精品高清在线| 久久精品成人欧美大片古装| 色播久久人人爽人人爽人人片视av| 粉嫩老牛aⅴ一区二区三区| 亚洲欧美日韩久久久久久| 红桃视频成人在线观看| 国产精品激情av电影在线观看| 久久久久久久久久久网站| 亚洲另类欧美自拍| 日韩亚洲第一页| 97视频网站入口| 欧美视频不卡中文| 国产精品久久久久久婷婷天堂| 中文字幕亚洲在线| 中文字幕日韩av综合精品| 91精品视频专区| 91牛牛免费视频| 欧美日韩成人在线观看| 日本高清久久天堂| 国产精品久久久久久久9999| 成人高清视频观看www| 国产日本欧美一区二区三区在线| 国产精品亚洲欧美导航| 亚洲精品不卡在线| www.日韩不卡电影av| 午夜精品久久久久久久99黑人| 国产精品麻豆va在线播放| 欧美激情18p| 国产91ⅴ在线精品免费观看| 欧美精品成人在线| 欧美理论在线观看| 在线国产精品播放| 91精品久久久久久综合乱菊| 国产精品一区二区性色av| 久久久久日韩精品久久久男男| 成人做爰www免费看视频网站| 日韩一二三在线视频播| 日韩电影中文字幕在线| 久久黄色av网站| 亚洲国产成人在线视频| 亚洲色图av在线| 亚洲男人天天操| 欧美成人小视频| 国产精品久久久久久久久久| 久久夜色精品国产欧美乱| 欧美日韩国产页| 国内精品小视频在线观看| 中文字幕精品在线| 国产精品第10页| 一区二区亚洲精品国产| 国产一区二区三区在线观看视频| 91色p视频在线| 韩国视频理论视频久久| 国产亚洲精品久久久久动| 91亚洲va在线va天堂va国| 另类图片亚洲另类| 日韩欧美国产黄色| 欧洲成人免费aa| 国产福利成人在线| 奇米四色中文综合久久| 欧美日韩免费在线| 亚洲第一在线视频| 日韩欧美中文第一页| 欧美大片免费观看在线观看网站推荐| 欧美激情区在线播放| 亚洲成av人乱码色午夜| 久久精品国产亚洲一区二区| 136fldh精品导航福利| 国产精品丝袜白浆摸在线| 色在人av网站天堂精品| 欧美视频中文字幕在线| 国产精品久久久久久搜索| 日韩三级成人av网| 亚洲va久久久噜噜噜久久天堂| 亚洲最新在线视频| 欧美高清第一页| 综合网日日天干夜夜久久| 久久国产精品久久国产精品| 日韩av片电影专区| 亚洲xxxxx电影| 亚洲第一免费播放区| 法国裸体一区二区| 日韩在线免费观看视频|