激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - Java多線程之線程池七個參數詳解

Java多線程之線程池七個參數詳解

2021-09-06 10:16一個抓手 Java教程

這篇文章主要介紹了Java多線程之線程池七個參數詳解,文中有很詳細的代碼示例,對正在學習java的小伙伴們有很好的幫助,需要的朋友可以參考下

ThreadPoolExecutor是JDK中的線程池實現,這個類實現了一個線程池需要的各個方法,它提供了任務提交、線程管理、監控等方法。

下面是ThreadPoolExecutor類的構造方法源碼,其他創建線程池的方法最終都會導向這個構造方法,共有7個參數:corePoolSize、maximumPoolSize、keepAliveTime、unit、workQueue、threadFactory、handler。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
public ThreadPoolExecutor(int corePoolSize,
                             int maximumPoolSize,
                             long keepAliveTime,
                             TimeUnit unit,
                             BlockingQueue<Runnable> workQueue,
                             ThreadFactory threadFactory,
                             RejectedExecutionHandler handler) {
       if (corePoolSize < 0 ||
           maximumPoolSize <= 0 ||
           maximumPoolSize < corePoolSize ||
           keepAliveTime < 0)
           throw new IllegalArgumentException();
       if (workQueue == null || threadFactory == null || handler == null)
           throw new NullPointerException();
       this.acc = System.getSecurityManager() == null ?
               null :
               AccessController.getContext();
       this.corePoolSize = corePoolSize;
       this.maximumPoolSize = maximumPoolSize;
       this.workQueue = workQueue;
       this.keepAliveTime = unit.toNanos(keepAliveTime);
       this.threadFactory = threadFactory;
       this.handler = handler;
   }

這些參數都通過volatile修飾:

?
1
2
3
4
5
6
7
8
9
10
public class ThreadPoolExecutor extends AbstractExecutorService {
    private final BlockingQueue<Runnable> workQueue;
    private volatile ThreadFactory threadFactory;
    private volatile RejectedExecutionHandler handler;
    private volatile long keepAliveTime;
    // 是否允許核心線程被回收
    private volatile boolean allowCoreThreadTimeOut;
    private volatile int corePoolSize;
    private volatile int maximumPoolSize;
}

corePoolSize:核心線程數

線程池維護的最小線程數量,核心線程創建后不會被回收(注意:設置allowCoreThreadTimeout=true后,空閑的核心線程超過存活時間也會被回收)。

大于核心線程數的線程,在空閑時間超過keepAliveTime后會被回收。

線程池剛創建時,里面沒有一個線程,當調用 execute() 方法添加一個任務時,如果正在運行的線程數量小于corePoolSize,則馬上創建新線程并運行這個任務。

maximumPoolSize:最大線程數

線程池允許創建的最大線程數量。

當添加一個任務時,核心線程數已滿,線程池還沒達到最大線程數,并且沒有空閑線程,工作隊列已滿的情況下,創建一個新線程,然后從工作隊列的頭部取出一個任務交由新線程來處理,而將剛提交的任務放入工作隊列尾部。

keepAliveTime:空閑線程存活時間

當一個可被回收的線程的空閑時間大于keepAliveTime,就會被回收。

可被回收的線程:

設置allowCoreThreadTimeout=true的核心線程。大于核心線程數的線程(非核心線程)。

unit:時間單位

keepAliveTime的時間單位:

?
1
2
3
4
5
6
7
TimeUnit.NANOSECONDS
TimeUnit.MICROSECONDS
TimeUnit.MILLISECONDS // 毫秒
TimeUnit.SECONDS
TimeUnit.MINUTES
TimeUnit.HOURS
TimeUnit.DAYS

workQueue:工作隊列

新任務被提交后,會先添加到工作隊列,任務調度時再從隊列中取出任務。工作隊列實現了BlockingQueue接口。

JDK默認的工作隊列有五種:

1.ArrayBlockingQueue 數組型阻塞隊列:數組結構,初始化時傳入大小,有界,FIFO,使用一個重入鎖,默認使用非公平鎖,入隊和出隊共用一個鎖,互斥。

2。LinkedBlockingQueue 鏈表型阻塞隊列:鏈表結構,默認初始化大小為Integer.MAX_VALUE,有界(近似無解),FIFO,使用兩個重入鎖分別控制元素的入隊和出隊,用Condition進行線程間的喚醒和等待。

3.SynchronousQueue 同步隊列:容量為0,添加任務必須等待取出任務,這個隊列相當于通道,不存儲元素。

4.PriorityBlockingQueue 優先阻塞隊列:無界,默認采用元素自然順序升序排列。

5.DelayQueue 延時隊列:無界,元素有過期時間,過期的元素才能被取出。

threadFactory:線程工廠

創建線程的工廠,可以設定線程名、線程編號等。

默認線程工廠:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
/**
   * The default thread factory
   */
  static class DefaultThreadFactory implements ThreadFactory {
      private static final AtomicInteger poolNumber = new AtomicInteger(1);
      private final ThreadGroup group;
      private final AtomicInteger threadNumber = new AtomicInteger(1);
      private final String namePrefix;
 
      DefaultThreadFactory() {
          SecurityManager s = System.getSecurityManager();
          group = (s != null) ? s.getThreadGroup() :
                                Thread.currentThread().getThreadGroup();
          namePrefix = "pool-" +
                        poolNumber.getAndIncrement() +
                       "-thread-";
      }
 
      public Thread newThread(Runnable r) {
          Thread t = new Thread(group, r,
                                namePrefix + threadNumber.getAndIncrement(),
                                0);
          if (t.isDaemon())
              t.setDaemon(false);
          if (t.getPriority() != Thread.NORM_PRIORITY)
              t.setPriority(Thread.NORM_PRIORITY);
          return t;
      }
  }

handler:拒絕策略

當線程池線程數已滿,并且工作隊列達到限制,新提交的任務使用拒絕策略處理。可以自定義拒絕策略,拒絕策略需要實現RejectedExecutionHandler接口。

JDK默認的拒絕策略有四種:

1.AbortPolicy:丟棄任務并拋出RejectedExecutionException異常。

2.DiscardPolicy:丟棄任務,但是不拋出異常。可能導致無法發現系統的異常狀態。

3.DiscardOldestPolicy:丟棄隊列最前面的任務,然后重新提交被拒絕的任務。

4.CallerRunsPolicy:由調用線程處理該任務。

默認拒絕策略:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
/**
   * The default rejected execution handler
   */
  private static final RejectedExecutionHandler defaultHandler = new AbortPolicy();
 
  public static class AbortPolicy implements RejectedExecutionHandler {
      /**
       * Creates an {@code AbortPolicy}.
       */
      public AbortPolicy() { }
 
      /**
       * Always throws RejectedExecutionException.
       *
       * @param r the runnable task requested to be executed
       * @param e the executor attempting to execute this task
       * @throws RejectedExecutionException always
       */
      public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {
          throw new RejectedExecutionException("Task " + r.toString() +
                                               " rejected from " +
                                               e.toString());
      }
  }

自定義線程池工具

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;
 
/**
 * 線程池工廠工具
 *
 * @author 向振華
 * @date 2021/04/11 10:24
 */
public class ThreadPoolFactory {
 
    /**
     * 生成固定大小的線程池
     *
     * @param threadName 線程名稱
     * @return 線程池
     */
    public static ExecutorService createFixedThreadPool(String threadName) {
        AtomicInteger threadNumber = new AtomicInteger(0);
        return new ThreadPoolExecutor(
                // 核心線程數
                desiredThreadNum(),
                // 最大線程數
                desiredThreadNum() * 2,
                // 空閑線程存活時間
                60L,
                // 空閑線程存活時間單位
                TimeUnit.SECONDS,
                // 工作隊列
                new ArrayBlockingQueue<>(1024),
                // 線程工廠
                new ThreadFactory() {
                    @Override
                    public Thread newThread(Runnable r) {
                        return new Thread(r, threadName + "-" + threadNumber.getAndIncrement());
                    }
                },
                // 拒絕策略
                new RejectedExecutionHandler() {
                    @Override
                    public void rejectedExecution(Runnable r, ThreadPoolExecutor executor) {
                        if (!executor.isShutdown()) {
                            try {
                                //嘗試阻塞式加入任務隊列
                                executor.getQueue().put(r);
                            } catch (Exception e) {
                                //保持線程的中斷狀態
                                Thread.currentThread().interrupt();
                            }
                        }
                    }
                });
    }
 
    /**
     * 理想的線程數,使用 2倍cpu核心數
     */
    public static int desiredThreadNum() {
        return Runtime.getRuntime().availableProcessors() * 2;
    }
}

到此這篇關于Java多線程之線程池七個參數詳解的文章就介紹到這了,更多相關java線程池詳解內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家!

原文鏈接:https://blog.csdn.net/Anenan/article/details/115603481

延伸 · 閱讀

精彩推薦
  • Java教程Java BufferWriter寫文件寫不進去或缺失數據的解決

    Java BufferWriter寫文件寫不進去或缺失數據的解決

    這篇文章主要介紹了Java BufferWriter寫文件寫不進去或缺失數據的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望...

    spcoder14552021-10-18
  • Java教程xml與Java對象的轉換詳解

    xml與Java對象的轉換詳解

    這篇文章主要介紹了xml與Java對象的轉換詳解的相關資料,需要的朋友可以參考下...

    Java教程網2942020-09-17
  • Java教程20個非常實用的Java程序代碼片段

    20個非常實用的Java程序代碼片段

    這篇文章主要為大家分享了20個非常實用的Java程序片段,對java開發項目有所幫助,感興趣的小伙伴們可以參考一下 ...

    lijiao5352020-04-06
  • Java教程Java實現搶紅包功能

    Java實現搶紅包功能

    這篇文章主要為大家詳細介紹了Java實現搶紅包功能,采用多線程模擬多人同時搶紅包,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙...

    littleschemer13532021-05-16
  • Java教程小米推送Java代碼

    小米推送Java代碼

    今天小編就為大家分享一篇關于小米推送Java代碼,小編覺得內容挺不錯的,現在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧...

    富貴穩中求8032021-07-12
  • Java教程Java8中Stream使用的一個注意事項

    Java8中Stream使用的一個注意事項

    最近在工作中發現了對于集合操作轉換的神器,java8新特性 stream,但在使用中遇到了一個非常重要的注意點,所以這篇文章主要給大家介紹了關于Java8中S...

    阿杜7472021-02-04
  • Java教程升級IDEA后Lombok不能使用的解決方法

    升級IDEA后Lombok不能使用的解決方法

    最近看到提示IDEA提示升級,尋思已經有好久沒有升過級了。升級完畢重啟之后,突然發現好多錯誤,本文就來介紹一下如何解決,感興趣的可以了解一下...

    程序猿DD9332021-10-08
  • Java教程Java使用SAX解析xml的示例

    Java使用SAX解析xml的示例

    這篇文章主要介紹了Java使用SAX解析xml的示例,幫助大家更好的理解和學習使用Java,感興趣的朋友可以了解下...

    大行者10067412021-08-30
主站蜘蛛池模板: 国产精品久久久久久久久久iiiii | 天天舔天天插 | 国产精品视频一区二区三区四 | 一级毛片在线观看视频 | 国产精品一区久久久久 | 毛片a片免费看 | 毛片免费在线观看视频 | 久久久99精品视频 | 亚洲精品 在线播放 | 国产精品一区二区在线 | 日本欧美一区二区三区在线播 | 色骚综合 | 毛片118极品美女写真 | 国产精品久久久久久久午夜片 | 亚洲福利视 | 成人国产视频在线观看 | 国产精品视频1区 | 成人区一区二区 | 久久精品视频12 | 成人在线观看一区二区三区 | 九九热在线视频免费观看 | 欧美成人精品一区二区 | av在线免费看网址 | 特级黄色一级毛片 | 欧美日韩亚洲在线 | 欧美精品videos | 在线成人av观看 | 一区二区免费看 | 一区二区三区欧美在线 | 欧美一级做一级爱a做片性 毛片电影网址 | 国产精品久久久久久久亚洲按摩 | 精品国产乱码久久久久久久 | lutube成人福利在线观看 | 在线成人免费观看www | 久久成人精品视频 | 狠狠干天天操 | 色屁屁xxxxⅹ免费视频 | 丰满年轻岳中文字幕一区二区 | 国产亚洲精品综合一区 | 深夜视频福利 | 成人国产精品一区 |