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

Java 線程池的作用以及該如何使用

 更新時(shí)間:2021年01月26日 09:04:06   作者:字母哥博客  
這篇文章主要介紹了Java 線程池的作用以及該如何使用,幫助大家更好的理解和學(xué)習(xí)Java的相關(guān)知識(shí),感興趣的朋友可以了解下

服務(wù)端應(yīng)用程序(如數(shù)據(jù)庫(kù)和 Web 服務(wù)器)需要處理來(lái)自客戶端的高并發(fā)、耗時(shí)較短的請(qǐng)求任務(wù),所以頻繁的創(chuàng)建處理這些請(qǐng)求的所需要的線程就是一個(gè)非常消耗資源的操作。常規(guī)的方法是針對(duì)一個(gè)新的請(qǐng)求創(chuàng)建一個(gè)新線程,雖然這種方法似乎易于實(shí)現(xiàn),但它有重大缺點(diǎn)。為每個(gè)請(qǐng)求創(chuàng)建新線程將花費(fèi)更多的時(shí)間,在創(chuàng)建和銷毀線程時(shí)花費(fèi)更多的系統(tǒng)資源。因此同時(shí)創(chuàng)建太多線程的 JVM 可能會(huì)導(dǎo)致系統(tǒng)內(nèi)存不足,這就需要限制要?jiǎng)?chuàng)建的線程數(shù),也就是需要使用到線程池。

一、什么是 Java 中的線程池?

線程池技術(shù)就是線程的重用技術(shù),使用之前創(chuàng)建好的線程來(lái)執(zhí)行當(dāng)前任務(wù),并提供了針對(duì)線程周期開銷和資源沖突問(wèn)題的解決方案。 由于請(qǐng)求到達(dá)時(shí)線程已經(jīng)存在,因此消除了線程創(chuàng)建過(guò)程導(dǎo)致的延遲,使應(yīng)用程序得到更快的響應(yīng)。

  • Java提供了以Executor接口及其子接口ExecutorService和ThreadPoolExecutor為中心的執(zhí)行器框架。通過(guò)使用Executor,完成線程任務(wù)只需實(shí)現(xiàn) Runnable接口并將其交給執(zhí)行器執(zhí)行即可。
  • 為您封裝好線程池,將您的編程任務(wù)側(cè)重于具體任務(wù)的實(shí)現(xiàn),而不是線程的實(shí)現(xiàn)機(jī)制。
  • 若要使用線程池,我們首先創(chuàng)建一個(gè) ExecutorService對(duì)象,然后向其傳遞一組任務(wù)。ThreadPoolExcutor 類則可以設(shè)置線程池初始化和最大的線程容量。

上圖表示線程池初始化具有3 個(gè)線程,任務(wù)隊(duì)列中有5 個(gè)待運(yùn)行的任務(wù)對(duì)象。

執(zhí)行器線程池方法

方法 描述
newFixedThreadPool(int) 創(chuàng)建具有固定的線程數(shù)的線程池,int參數(shù)表示線程池內(nèi)線程的數(shù)量
newCachedThreadPool() 創(chuàng)建一個(gè)可緩存線程池,該線程池可靈活回收空閑線程。若無(wú)空閑線程,則新建線程處理任務(wù)。
newSingleThreadExecutor() 創(chuàng)建一個(gè)單線程化的線程池,它只會(huì)用唯一的工作線程來(lái)執(zhí)行任務(wù)
newScheduledThreadPool 創(chuàng)建一個(gè)定長(zhǎng)線程池,支持定時(shí)及周期性任務(wù)執(zhí)行

在固定線程池的情況下,如果執(zhí)行器當(dāng)前運(yùn)行的所有線程,則掛起的任務(wù)將放在隊(duì)列中,并在線程變?yōu)榭臻e時(shí)執(zhí)行。

二、線程池示例

在下面的內(nèi)容中,我們將介紹線程池的executor執(zhí)行器。

創(chuàng)建線程池處理任務(wù)要遵循的步驟

  1. 創(chuàng)建一個(gè)任務(wù)對(duì)象(實(shí)現(xiàn)Runnable接口),用于執(zhí)行具體的任務(wù)邏輯
  2. 使用Executors創(chuàng)建線程池ExecutorService
  3. 將待執(zhí)行的任務(wù)對(duì)象交給ExecutorService進(jìn)行任務(wù)處理
  4. 停掉 Executor 線程池
//第一步: 創(chuàng)建一個(gè)任務(wù)對(duì)象(實(shí)現(xiàn)Runnable接口),用于執(zhí)行具體的任務(wù)邏輯 (Step 1) 
class Task implements Runnable {
  private String name;

  public Task(String s) {
    name = s;
  }

  // 打印任務(wù)名稱并Sleep 1秒
  // 整個(gè)處理流程執(zhí)行5次
  public void run() {
    try{
      for (int i = 0; i<=5; i++) {
        if (i==0) {
          Date d = new Date();
          SimpleDateFormat ft = new SimpleDateFormat("hh:mm:ss");
          System.out.println("任務(wù)初始化" + name +" = " + ft.format(d));
          //第一次執(zhí)行的時(shí)候,打印每一個(gè)任務(wù)的名稱及初始化的時(shí)間
        }
        else{
          Date d = new Date();
          SimpleDateFormat ft = new SimpleDateFormat("hh:mm:ss");
          System.out.println("任務(wù)正在執(zhí)行" + name +" = " + ft.format(d));
          // 打印每一個(gè)任務(wù)處理的執(zhí)行時(shí)間
        }
        Thread.sleep(1000);
      }
      System.out.println("任務(wù)執(zhí)行完成" + name);
    } catch(InterruptedException e) {
      e.printStackTrace();
    }
  }
}

測(cè)試用例

public class ThreadPoolTest {
  // 線程池里面最大線程數(shù)量
  static final int MAX_SIZE = 3;

  public static void main (String[] args) {
    // 創(chuàng)建5個(gè)任務(wù)
    Runnable r1 = new Task("task 1");
    Runnable r2 = new Task("task 2");
    Runnable r3 = new Task("task 3");
    Runnable r4 = new Task("task 4");
    Runnable r5 = new Task("task 5");

    // 第二步:創(chuàng)建一個(gè)固定線程數(shù)量的線程池,線程數(shù)為MAX_SIZE
    ExecutorService pool = Executors.newFixedThreadPool(MAX_SIZE);

    // 第三步:將待執(zhí)行的任務(wù)對(duì)象交給ExecutorService進(jìn)行任務(wù)處理
    pool.execute(r1);
    pool.execute(r2);
    pool.execute(r3);
    pool.execute(r4);
    pool.execute(r5);

    // 第四步:關(guān)閉線程池
    pool.shutdown();
  }
} 

示例執(zhí)行結(jié)果

任務(wù)初始化task 1 = 05:25:55
任務(wù)初始化task 2 = 05:25:55
任務(wù)初始化task 3 = 05:25:55
任務(wù)正在執(zhí)行task 3 = 05:25:56
任務(wù)正在執(zhí)行task 1 = 05:25:56
任務(wù)正在執(zhí)行task 2 = 05:25:56
任務(wù)正在執(zhí)行task 1 = 05:25:57
任務(wù)正在執(zhí)行task 3 = 05:25:57
任務(wù)正在執(zhí)行task 2 = 05:25:57
任務(wù)正在執(zhí)行task 3 = 05:25:58
任務(wù)正在執(zhí)行task 1 = 05:25:58
任務(wù)正在執(zhí)行task 2 = 05:25:58
任務(wù)正在執(zhí)行task 2 = 05:25:59
任務(wù)正在執(zhí)行task 3 = 05:25:59
任務(wù)正在執(zhí)行task 1 = 05:25:59
任務(wù)正在執(zhí)行task 1 = 05:26:00
任務(wù)正在執(zhí)行task 2 = 05:26:00
任務(wù)正在執(zhí)行task 3 = 05:26:00
任務(wù)執(zhí)行完成task 3
任務(wù)執(zhí)行完成task 2
任務(wù)執(zhí)行完成task 1
任務(wù)初始化task 5 = 05:26:01
任務(wù)初始化task 4 = 05:26:01
任務(wù)正在執(zhí)行task 4 = 05:26:02
任務(wù)正在執(zhí)行task 5 = 05:26:02
任務(wù)正在執(zhí)行task 4 = 05:26:03
任務(wù)正在執(zhí)行task 5 = 05:26:03
任務(wù)正在執(zhí)行task 5 = 05:26:04
任務(wù)正在執(zhí)行task 4 = 05:26:04
任務(wù)正在執(zhí)行task 4 = 05:26:05
任務(wù)正在執(zhí)行task 5 = 05:26:05
任務(wù)正在執(zhí)行task 4 = 05:26:06
任務(wù)正在執(zhí)行task 5 = 05:26:06
任務(wù)執(zhí)行完成task 4
任務(wù)執(zhí)行完成task 5

如程序執(zhí)行結(jié)果中顯示的一樣,任務(wù) 4 或任務(wù) 5 僅在池中的線程變?yōu)榭臻e時(shí)才執(zhí)行。在此之前,額外的任務(wù)將放在待執(zhí)行的隊(duì)列中。

線程池執(zhí)行前三個(gè)任務(wù),線程池內(nèi)線程回收空出來(lái)之后再去處理執(zhí)行任務(wù) 4 和 5

使用這種線程池方法的一個(gè)主要優(yōu)點(diǎn)是,假如您希望一次處理10000個(gè)請(qǐng)求,但不希望創(chuàng)建10000個(gè)線程,從而避免造成系統(tǒng)資源的過(guò)量使用導(dǎo)致的宕機(jī)。您可以使用此方法創(chuàng)建一個(gè)包含500個(gè)線程的線程池,并且可以向該線程池提交500個(gè)請(qǐng)求。
ThreadPool此時(shí)將創(chuàng)建最多500個(gè)線程,一次處理500個(gè)請(qǐng)求。在任何一個(gè)線程的進(jìn)程完成之后,ThreadPool將在內(nèi)部將第501個(gè)請(qǐng)求分配給該線程,并將繼續(xù)對(duì)所有剩余的請(qǐng)求執(zhí)行相同的操作。在系統(tǒng)資源比較緊張的情況下,線程池是保證程序穩(wěn)定運(yùn)行的一個(gè)有效的解決方案。

三、使用線程池的注意事項(xiàng)與調(diào)優(yōu)

  1. 死鎖: 雖然死鎖可能發(fā)生在任何多線程程序中,但線程池引入了另一個(gè)死鎖案例,其中所有執(zhí)行線程都在等待隊(duì)列中某個(gè)阻塞線程的執(zhí)行結(jié)果,導(dǎo)致線程無(wú)法繼續(xù)執(zhí)行。
  2. 線程泄漏 : 如果線程池中線程在任務(wù)完成時(shí)未正確返回,將發(fā)生線程泄漏問(wèn)題。例如,某個(gè)線程引發(fā)異常并且池類沒(méi)有捕獲此異常,則線程將異常退出,從而線程池的大小將減小一個(gè)。如果這種情況重復(fù)多次,則線程池最終將變?yōu)榭?,沒(méi)有線程可用于執(zhí)行其他任務(wù)。
  3. 線程頻繁輪換: 如果線程池大小非常大,則線程之間進(jìn)行上下文切換會(huì)浪費(fèi)很多時(shí)間。所以在系統(tǒng)資源允許的情況下,也不是線程池越大越好。

線程池大小優(yōu)化: 線程池的最佳大小取決于可用的處理器數(shù)量和待處理任務(wù)的性質(zhì)。對(duì)于CPU密集型任務(wù),假設(shè)系統(tǒng)有N個(gè)邏輯處理核心,N 或 N+1 的最大線程池?cái)?shù)量大小將實(shí)現(xiàn)最大效率。對(duì)于 I/O密集型任務(wù),需要考慮請(qǐng)求的等待時(shí)間(W)和服務(wù)處理時(shí)間(S)的比例,線程池最大大小為 N*(1+ W/S)會(huì)實(shí)現(xiàn)最高效率。

不要教條的使用上面的總結(jié),需要根據(jù)自己的應(yīng)用任務(wù)處理類型進(jìn)行靈活的設(shè)置與調(diào)優(yōu),其中少不了測(cè)試實(shí)驗(yàn)。

原文鏈接:字母哥博客。

以上就是Java 線程池的作用以及該如何使用的詳細(xì)內(nèi)容,更多關(guān)于Java 線程池的作用和使用的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • java JDBC系列教程之JDBC類的簡(jiǎn)析與JDBC的基礎(chǔ)操作

    java JDBC系列教程之JDBC類的簡(jiǎn)析與JDBC的基礎(chǔ)操作

    這篇文章主要介紹了java JDBC系列教程之JDBC類的簡(jiǎn)析與JDBC的基礎(chǔ)操作,本文分步驟通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-07-07
  • springboot swagger 接口文檔分組展示功能實(shí)現(xiàn)

    springboot swagger 接口文檔分組展示功能實(shí)現(xiàn)

    這篇文章主要介紹了springboot swagger 接口文檔分組展示功能實(shí)現(xiàn),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),感興趣的朋友跟隨小編一起看看吧
    2024-03-03
  • java高并發(fā)的并發(fā)級(jí)別詳解

    java高并發(fā)的并發(fā)級(jí)別詳解

    這篇文章主要介紹了java高并發(fā)的并發(fā)級(jí)別,內(nèi)容十分豐富,在這里分享給大家,需要的朋友可以參考,希望能夠給你帶來(lái)幫助
    2021-10-10
  • idea中無(wú)法自動(dòng)裝配未找到 ‘XXXXXXX‘ 類型的 Bean

    idea中無(wú)法自動(dòng)裝配未找到 ‘XXXXXXX‘ 類型的 Bean

    本文主要介紹了idea中無(wú)法自動(dòng)裝配未找到 ‘XXXXXXX‘ 類型的 Bean的原因及三種解決方法,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-03-03
  • springboot全局配置文件與多環(huán)境配置的全過(guò)程

    springboot全局配置文件與多環(huán)境配置的全過(guò)程

    SpringBoot項(xiàng)目在多環(huán)境配置上表現(xiàn)的非常優(yōu)秀,只需要非常簡(jiǎn)單的操作就可以完成配置,下面這篇文章主要給大家介紹了關(guān)于springboot全局配置文件與多環(huán)境配置的相關(guān)資料,需要的朋友可以參考下
    2021-12-12
  • 使用Java實(shí)現(xiàn)驗(yàn)證碼程序

    使用Java實(shí)現(xiàn)驗(yàn)證碼程序

    這篇文章主要為大家詳細(xì)介紹了使用Java實(shí)現(xiàn)驗(yàn)證碼程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • Java用for循環(huán)Map詳細(xì)解析

    Java用for循環(huán)Map詳細(xì)解析

    本篇文章主要介紹了Java用for循環(huán)Map,需要的朋友可以過(guò)來(lái)參考下,希望對(duì)大家有所幫助
    2013-12-12
  • 常見(jiàn)的排序算法,一篇就夠了

    常見(jiàn)的排序算法,一篇就夠了

    這篇文章主要介紹了一些常用排序算法整理,插入排序算法、直接插入排序、希爾排序、選擇排序、冒泡排序等排序,需要的朋友可以參考下
    2021-07-07
  • SpringBoot操作Jedis案例代碼

    SpringBoot操作Jedis案例代碼

    這篇文章主要介紹了SpringBoot操作Jedis案例代碼,代碼部分包括pom依賴、配置相關(guān)參數(shù)、JedisPool的設(shè)置,代碼簡(jiǎn)單易懂對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-08-08
  • Java使用線程池實(shí)現(xiàn)socket編程的方法詳解

    Java使用線程池實(shí)現(xiàn)socket編程的方法詳解

    這篇文章主要為大家詳細(xì)介紹了Java使用線程池實(shí)現(xiàn)socket編程的方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-03-03

最新評(píng)論