線程池是什么?Java四種線程池的使用介紹


使用線程池的好處有很多,比如節省系統資源的開銷,節省創建和銷毀線程的時間等,當我們需要處理的任務較多時,就可以使用線程池,可能還有很多用戶不知道Java線程池如何使用?下面小編給大家分享Java四種線程池的使用方法。

  線程池介紹:

  線程池是一種多線程處理形式,處理過程中將任務添加到隊列,然后在創建線程后自動啟動這些任務。線程池線程都是后台線程。每個線程都使用默認的堆棧大小,以默認的優先級運行,並處於多線程單元中。如果某個線程在托管代碼中空閑(如正在等待某個事件),則線程池將插入另一個輔助線程來使所有處理器保持繁忙。如果所有線程池線程都始終保持繁忙,但隊列中包含掛起的工作,則線程池將在一段時間后創建另一個輔助線程但線程的數目永遠不會超過最大值。超過最大值的線程可以排隊,但他們要等到其他線程完成后才啟動。

  Java四種線程池的使用:

  Java通過Executors提供四種線程池,分別為:

  newCachedThreadPool創建一個可緩存線程池,如果線程池長度超過處理需要,可靈活回收空閑線程,若無可回收,則新建線程。

  newFixedThreadPool 創建一個定長線程池,可控制線程最大並發數,超出的線程會在隊列中等待。

  newScheduledThreadPool 創建一個定長線程池,支持定時及周期性任務執行。

  newSingleThreadExecutor 創建一個單線程化的線程池,它只會用唯一的工作線程來執行任務,保證所有任務按照指定順序(FIFO, LIFO, 優先級)執行。

  (1) newCachedThreadPool

  創建一個可緩存線程池,如果線程池長度超過處理需要,可靈活回收空閑線程,若無可回收,則新建線程。示例代碼如下:

  1. 01package test;
  2. 02import java.util.concurrent.ExecutorService;
  3. 03import java.util.concurrent.Executors;
  4. 04public class ThreadPoolExecutorTest {
  5. 05public static void main(String[] args) {
  6. 06ExecutorService cachedThreadPool = Executors.newCachedThreadPool();
  7. 07for (int i = 0; i < 10; i++) {
  8. 08final int index = i;
  9. 09try {
  10. 10Thread.sleep(index * 1000);
  11. 11} catch (InterruptedException e) {
  12. 12e.printStackTrace();
  13. 13}
  14. 14cachedThreadPool.execute(new Runnable() {
  15. 15public void run() {
  16. 16System.out.println(index);
  17. 17}
  18. 18});
  19. 19}
  20. 20}
  21. 21}
復制代碼

  線程池為無限大,當執行第二個任務時第一個任務已經完成,會復用執行第一個任務的線程,而不用每次新建線程。

  (2) newFixedThreadPool

  創建一個定長線程池,可控制線程最大並發數,超出的線程會在隊列中等待。示例代碼如下:

  1. 01package test;
  2. 02import java.util.concurrent.ExecutorService;
  3. 03import java.util.concurrent.Executors;
  4. 04public class ThreadPoolExecutorTest {
  5. 05public static void main(String[] args) {
  6. 06ExecutorService fixedThreadPool = Executors.newFixedThreadPool(3);
  7. 07for (int i = 0; i < 10; i++) {
  8. 08final int index = i;
  9. 09fixedThreadPool.execute(new Runnable() {
  10. 10public void run() {
  11. 11try {
  12. 12System.out.println(index);
  13. 13Thread.sleep(2000);
  14. 14} catch (InterruptedException e) {
  15. 15e.printStackTrace();
  16. 16}
  17. 17}
  18. 18});
  19. 19}
  20. 20}
  21. 21}
復制代碼

  因為線程池大小為3,每個任務輸出index后sleep 2秒,所以每兩秒打印3個數字。

  定長線程池的大小最好根據系統資源進行設置。如Runtime.getRuntime().availableProcessors()

  (3) newScheduledThreadPool

  創建一個定長線程池,支持定時及周期性任務執行。延遲執行示例代碼如下:

  1. 01package test;
  2. 02import java.util.concurrent.Executors;
  3. 03import java.util.concurrent.ScheduledExecutorService;
  4. 04import java.util.concurrent.TimeUnit;
  5. 05public class ThreadPoolExecutorTest {
  6. 06public static void main(String[] args) {
  7. 07ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(5);
  8. 08scheduledThreadPool.schedule(new Runnable() {
  9. 09public void run() {
  10. 10System.out.println("delay 3 seconds");
  11. 11}
  12. 12}, 3, TimeUnit.SECONDS);
  13. 13}
  14. 14}
復制代碼

  表示延遲3秒執行。

  定期執行示例代碼如下:

  1. 01package test;
  2. 02import java.util.concurrent.Executors;
  3. 03import java.util.concurrent.ScheduledExecutorService;
  4. 04import java.util.concurrent.TimeUnit;
  5. 05public class ThreadPoolExecutorTest {
  6. 06public static void main(String[] args) {
  7. 07ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(5);
  8. 08scheduledThreadPool.scheduleAtFixedRate(new Runnable() {
  9. 09public void run() {
  10. 10System.out.println("delay 1 seconds, and excute every 3 seconds");
  11. 11}
  12. 12}, 1, 3, TimeUnit.SECONDS);
  13. 13}
  14. 14}
復制代碼

  表示延遲1秒后每3秒執行一次。

  (4) newSingleThreadExecutor

  創建一個單線程化的線程池,它只會用唯一的工作線程來執行任務,保證所有任務按照指定順序(FIFO, LIFO, 優先級)執行。示例代碼如下:

  1. 01package test;
  2. 02import java.util.concurrent.ExecutorService;
  3. 03import java.util.concurrent.Executors;
  4. 04public class ThreadPoolExecutorTest {
  5. 05public static void main(String[] args) {
  6. 06ExecutorService singleThreadExecutor = Executors.newSingleThreadExecutor();
  7. 07for (int i = 0; i < 10; i++) {
  8. 08final int index = i;
  9. 09singleThreadExecutor.execute(new Runnable() {
  10. 10public void run() {
  11. 11try {
  12. 12System.out.println(index);
  13. 13Thread.sleep(2000);
  14. 14} catch (InterruptedException e) {
  15. 15e.printStackTrace();
  16. 16}
  17. 17}
  18. 18});
  19. 19}
  20. 20}
  21. 21}
復制代碼

  結果依次輸出,相當於順序執行各個任務。

  你可以使用JDK自帶的監控工具來監控我們創建的線程數量,運行一個不終止的線程,創建指定量的線程,來觀察:

  工具目錄:C:\Program Files\Java\jdk1.6.0_06\bin\jconsole.exe

  運行程序做稍微修改:

  1. 01package test;
  2. 02import java.util.concurrent.ExecutorService;
  3. 03import java.util.concurrent.Executors;
  4. 04public class ThreadPoolExecutorTest {
  5. 05public static void main(String[] args) {
  6. 06ExecutorService singleThreadExecutor = Executors.newCachedThreadPool();
  7. 07for (int i = 0; i < 100; i++) {
  8. 08final int index = i;
  9. 09singleThreadExecutor.execute(new Runnable() {
  10. 10public void run() {
  11. 11try {
  12. 12while(true) {
  13. 13System.out.println(index);
  14. 14Thread.sleep(10 * 1000);
  15. 15}
  16. 16} catch (InterruptedException e) {
  17. 17e.printStackTrace();
  18. 18}
  19. 19}
  20. 20});
  21. 21try {
  22. 22Thread.sleep(500);
  23. 23} catch (InterruptedException e) {
  24. 24e.printStackTrace();
  25. 25}
  26. 26}
  27. 27}
  28. 28}

 

監控運行狀態

  關於Java四種線程池的使用技巧就給大家分享到這里了,正所謂工欲善其事,必先利其器,我們掌握了訣竅,處理事情才能事倍功半,希望可以幫助到大家。

 

如有不足之處請諒解:寫下你寶貴的留言,我會及時改正

----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

偉大人nice

E-mail :dawei_818@163.com

QQ :602091999

Cell-phone number :15800666248 

http://www.cnblogs.com/lsw9/

  歡迎騷擾  

如有不足之處請諒解:寫下你寶貴的留言,我會及時改正

----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM