为什么用线程池

原文地址 http://blog.csdn.net/qq_25806863/article/details/71126867

有时候,系统需要处理非常多的执行时间很短的请求,如果每一个请求都开启一个新线程的话,系统就要不断的进行线程的创建和销毁,有时花在创建和销毁线程上的时间会比线程真正执行的时间还长。而且当线程数量太多时,系统不一定能受得了。

使用线程池主要为了解决一下几个问题:

通过重用线程池中的线程,来减少每个线程创建和销毁的性能开销。
对线程进行一些维护和管理,比如定时开始,周期执行,并发数控制等等。
Executor
Executor是一个接口,跟线程池有关的基本都要跟他打交道。下面是常用的ThreadPoolExecutor的关系。

Executor接口很简单,只有一个execute方法。

ExecutorService是Executor的子接口,增加了一些常用的对线程的控制方法,之后使用线程池主要也是使用这些方法。

AbstractExecutorService是一个抽象类。ThreadPoolExecutor就是实现了这个类。

ThreadPoolExecutor
构造方法
ThreadPoolExecutor是线程池的真正实现,他通过构造方法的一系列参数,来构成不同配置的线程池。常用的构造方法有下面四个:

  1. ThreadPoolExecutor(int corePoolSize,
  2. int maximumPoolSize,
  3. long keepAliveTime,
  4. TimeUnit unit,
  5. BlockingQueue<Runnable> workQueue)
  6.  
  7. ThreadPoolExecutor(int corePoolSize,
  8. int maximumPoolSize,
  9. long keepAliveTime,
  10. TimeUnit unit,
  11. BlockingQueue<Runnable> workQueue,
  12. ThreadFactory threadFactory)
  13.  
  14. ThreadPoolExecutor(int corePoolSize,
  15. int maximumPoolSize,
  16. long keepAliveTime,
  17. TimeUnit unit,
  18. BlockingQueue<Runnable> workQueue,
  19. RejectedExecutionHandler handler)
  20.  
  21. ThreadPoolExecutor(int corePoolSize,
  22. int maximumPoolSize,
  23. long keepAliveTime,
  24. TimeUnit unit,
  25. BlockingQueue<Runnable> workQueue,
  26. ThreadFactory threadFactory,
  27. RejectedExecutionHandler handler)

构造方法参数说明

  • corePoolSize

  核心线程数,默认情况下核心线程会一直存活,即使处于闲置状态也不会受存keepAliveTime限制。除非将allowCoreThreadTimeOut设置为true。

  • maximumPoolSize

  线程池所能容纳的最大线程数。超过这个数的线程将被阻塞。当任务队列为没有设置大小的LinkedBlockingDeque时,这个值无效。

  • keepAliveTime

  非核心线程的闲置超时时间,超过这个时间就会被回收。

  • unit

  指定keepAliveTime的单位,如TimeUnit.SECONDS。当将allowCoreThreadTimeOut设置为true时对corePoolSize生效。

  • workQueue

线程池中的任务队列.

常用的有三种队列,SynchronousQueue,LinkedBlockingDeque,ArrayBlockingQueue。

threadFactory

线程工厂,提供创建新线程的功能。ThreadFactory是一个接口,只有一个方法

  1. public interface ThreadFactory {
  2.   Thread newThread(Runnable r);
  3. }

通过线程工厂可以对线程的一些属性进行定制。

默认的工厂:

  1. static class DefaultThreadFactory implements ThreadFactory {
  2. private static final AtomicInteger poolNumber = new AtomicInteger(1);
  3. private final ThreadGroup group;
  4. private final AtomicInteger threadNumber = new AtomicInteger(1);
  5. private final String namePrefix;
  6.  
  7. DefaultThreadFactory() {
  8. SecurityManager var1 = System.getSecurityManager();
  9. this.group = var1 != null?var1.getThreadGroup():Thread.currentThread().getThreadGroup();
  10. this.namePrefix = "pool-" + poolNumber.getAndIncrement() + "-thread-";
  11. }
  12.  
  13. public Thread newThread(Runnable var1) {
  14. Thread var2 = new Thread(this.group, var1, this.namePrefix + this.threadNumber.getAndIncrement(), 0L);
  15. if(var2.isDaemon()) {
  16. var2.setDaemon(false);
  17. }
  18.  
  19. if(var2.getPriority() != 5) {
  20. var2.setPriority(5);
  21. }
  22.  
  23. return var2;
  24. }
  25. }
  • RejectedExecutionHandler

  RejectedExecutionHandler也是一个接口,只有一个方法

  1. public interface RejectedExecutionHandler {
  2. void rejectedExecution(Runnable var1, ThreadPoolExecutor var2);
  3. }

当线程池中的资源已经全部使用,添加新线程被拒绝时,会调用RejectedExecutionHandler的rejectedExecution方法。

线程池规则
线程池的线程执行规则跟任务队列有很大的关系。

  • 下面都假设任务队列没有大小限制
  1. 如果线程数量<=核心线程数量,那么直接启动一个核心线程来执行任务,不会放入队列中。
  2. 如果线程数量>核心线程数,但<=最大线程数,并且任务队列是LinkedBlockingDeque的时候,超过核心线程数量的任务会放在任务队列中排队。
  3. 如果线程数量>核心线程数,但<=最大线程数,并且任务队列是SynchronousQueue的时候,线程池会创建新线程执行任务,这些任务也不会被放在任务队列中。这些线程属于非核心线程,在任务完成后,闲置时间达到了超时时间就会被清除。
  4. 如果线程数量>核心线程数,并且>最大线程数,当任务队列是LinkedBlockingDeque,会将超过核心线程的任务放在任务队列中排队。也就是当任务队列是LinkedBlockingDeque并且没有大小限制时,线程池的最大线程数设置是无效的,他的线程数最多不会超过核心线程数。
  5. 如果线程数量>核心线程数,并且>最大线程数,当任务队列是SynchronousQueue的时候,会因为线程池拒绝添加任务而抛出异常。
  • 任务队列大小有限时
  1. 当LinkedBlockingDeque塞满时,新增的任务会直接创建新线程来执行,当创建的线程数量超过最大线程数量时会抛异常。
  2. SynchronousQueue没有数量限制。因为他根本不保持这些任务,而是直接交给线程池去执行。当任务数量超过最大线程数时会直接抛异常。

规则验证
前提
所有的任务都是下面这样的,睡眠两秒后打印一行日志:

  1. Runnable myRunnable = new Runnable() {
  2. @Override
  3. public void run() {
  4. try {
  5. Thread.sleep(2000);
  6. System.out.println(Thread.currentThread().getName() + " run");
  7. } catch (InterruptedException e) {
  8. e.printStackTrace();
  9. }
  10.  
  11. }
  12. };

所有验证过程都是下面这样,先执行三个,再执行三个,8秒后,各看一次信息

  1. executor.execute(myRunnable);
  2. executor.execute(myRunnable);
  3. executor.execute(myRunnable);
  4. System.out.println("---先开三个---");
  5. System.out.println("核心线程数" + executor.getCorePoolSize());
  6. System.out.println("线程池数" + executor.getPoolSize());
  7. System.out.println("队列任务数" + executor.getQueue().size());
  8. executor.execute(myRunnable);
  9. executor.execute(myRunnable);
  10. executor.execute(myRunnable);
  11. System.out.println("---再开三个---");
  12. System.out.println("核心线程数" + executor.getCorePoolSize());
  13. System.out.println("线程池数" + executor.getPoolSize());
  14. System.out.println("队列任务数" + executor.getQueue().size());
  15. Thread.sleep(8000);
  16. System.out.println("----8秒之后----");
  17. System.out.println("核心线程数" + executor.getCorePoolSize());
  18. System.out.println("线程池数" + executor.getPoolSize());
  19. System.out.println("队列任务数" + executor.getQueue().size());

验证1

核心线程数为6,最大线程数为10。超时时间为5秒

  1. ThreadPoolExecutor executor = new ThreadPoolExecutor(6, 10, 5, TimeUnit.SECONDS, new SynchronousQueue<Runnable>());
  1. ---先开三个---
  2. 核心线程数6
  3. 线程池线程数3
  4. 队列任务数0
  5. ---再开三个---
  6. 核心线程数6
  7. 线程池线程数6
  8. 队列任务数0
  9. pool-1-thread-1 run
  10. pool-1-thread-6 run
  11. pool-1-thread-5 run
  12. pool-1-thread-3 run
  13. pool-1-thread-4 run
  14. pool-1-thread-2 run
  15. ----8秒之后----
  16. 核心线程数6
  17. 线程池线程数6
  18. 队列任务数0

可以看到每个任务都是是直接启动一个核心线程来执行任务,一共创建了6个线程,不会放入队列中。8秒后线程池还是6个线程,核心线程默认情况下不会被回收,不收超时时间限制。

验证2
核心线程数为3,最大线程数为6。超时时间为5秒,队列是LinkedBlockingDeque

  1. ThreadPoolExecutor executor = new ThreadPoolExecutor(3, 6, 5, TimeUnit.SECONDS, new LinkedBlockingDeque<Runnable>());
  1. ---先开三个---
  2. 核心线程数3
  3. 线程池线程数3
  4. 队列任务数0
  5. ---再开三个---
  6. 核心线程数3
  7. 线程池线程数3
  8. 队列任务数3
  9. pool-1-thread-3 run
  10. pool-1-thread-1 run
  11. pool-1-thread-2 run
  12. pool-1-thread-3 run
  13. pool-1-thread-1 run
  14. pool-1-thread-2 run
  15. ----8秒之后----
  16. 核心线程数3
  17. 线程池线程数3
  18. 队列任务数0

当任务数超过核心线程数时,会将超出的任务放在队列中,只会创建3个线程重复利用。

验证3
核心线程数为3,最大线程数为6。超时时间为5秒,队列是SynchronousQueue

  1. ThreadPoolExecutor executor = new ThreadPoolExecutor(3, 6, 5, TimeUnit.SECONDS, new SynchronousQueue<Runnable>());
  1. ---先开三个---
  2. 核心线程数3
  3. 线程池线程数3
  4. 队列任务数0
  5. ---再开三个---
  6. 核心线程数3
  7. 线程池线程数6
  8. 队列任务数0
  9. pool-1-thread-2 run
  10. pool-1-thread-3 run
  11. pool-1-thread-6 run
  12. pool-1-thread-4 run
  13. pool-1-thread-5 run
  14. pool-1-thread-1 run
  15. ----8秒之后----
  16. 核心线程数3
  17. 线程池线程数3
  18. 队列任务数0

当队列是SynchronousQueue时,超出核心线程的任务会创建新的线程来执行,看到一共有6个线程。但是这些线程是费核心线程,收超时时间限制,在任务完成后限制超过5秒就会被回收。所以最后看到线程池还是只有三个线程。

验证4
核心线程数是3,最大线程数是4,队列是LinkedBlockingDeque

  1. ThreadPoolExecutor executor = new ThreadPoolExecutor(3, 4, 5, TimeUnit.SECONDS, new LinkedBlockingDeque<Runnable>());
  1. ---先开三个---
  2. 核心线程数3
  3. 线程池线程数3
  4. 队列任务数0
  5. ---再开三个---
  6. 核心线程数3
  7. 线程池线程数3
  8. 队列任务数3
  9. pool-1-thread-3 run
  10. pool-1-thread-1 run
  11. pool-1-thread-2 run
  12. pool-1-thread-3 run
  13. pool-1-thread-1 run
  14. pool-1-thread-2 run
  15. ----8秒之后----
  16. 核心线程数3
  17. 线程池线程数3
  18. 队列任务数0

LinkedBlockingDeque根本不受最大线程数影响。

但是当LinkedBlockingDeque有大小限制时就会受最大线程数影响了

4.1 比如下面,将队列大小设置为2.

  1. ThreadPoolExecutor executor = new ThreadPoolExecutor(3, 4, 5, TimeUnit.SECONDS, new LinkedBlockingDeque<Runnable>(2));
  1. ---先开三个---
  2. 核心线程数3
  3. 线程池线程数3
  4. 队列任务数0
  5. ---再开三个---
  6. 核心线程数3
  7. 线程池线程数4
  8. 队列任务数2
  9. pool-1-thread-2 run
  10. pool-1-thread-1 run
  11. pool-1-thread-4 run
  12. pool-1-thread-3 run
  13. pool-1-thread-1 run
  14. pool-1-thread-2 run
  15. ----8秒之后----
  16. 核心线程数3
  17. 线程池线程数3
  18. 队列任务数0

首先为三个任务开启了三个核心线程1,2,3,然后第四个任务和第五个任务加入到队列中,第六个任务因为队列满了,就直接创建一个新线程4,这是一共有四个线程,没有超过最大线程数。8秒后,非核心线程收超时时间影响回收了,因此线程池只剩3个线程了。

4.2 将队列大小设置为1

  1. ThreadPoolExecutor executor = new ThreadPoolExecutor(3, 4, 5, TimeUnit.SECONDS, new LinkedBlockingDeque<Runnable>(1));
  1. Exception in thread "main" java.util.concurrent.RejectedExecutionException: Task com.sunlinlin.threaddemo.Main$1@677327b6 rejected from java.util.concurrent.ThreadPoolExecutor@14ae5a5[Running, pool size = 4, active threads = 4, queued tasks = 1, completed tasks = 0]
  2. at java.util.concurrent.ThreadPoolExecutor$AbortPolicy.rejectedExecution(ThreadPoolExecutor.java:2047)
  3. at java.util.concurrent.ThreadPoolExecutor.reject(ThreadPoolExecutor.java:823)
  4. at java.util.concurrent.ThreadPoolExecutor.execute(ThreadPoolExecutor.java:1369)
  5. at com.sunlinlin.threaddemo.Main.main(Main.java:35)
  6. at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
  7. at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
  8. at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
  9. at java.lang.reflect.Method.invoke(Method.java:498)
  10. at com.intellij.rt.execution.application.AppMain.main(AppMain.java:147)
  11. ---先开三个---
  12. 核心线程数3
  13. 线程池线程数3
  14. 队列任务数0
  15. pool-1-thread-1 run
  16. pool-1-thread-2 run
  17. pool-1-thread-3 run
  18. pool-1-thread-4 run
  19. pool-1-thread-1 run

直接出错在第6个execute方法上。因为核心线程是3个,当加入第四个任务的时候,就把第四个放在队列中。加入第五个任务时,因为队列满了,就创建新线程执行,创建了线程4。当加入第六个线程时,也会尝试创建线程,但是因为已经达到了线程池最大线程数,所以直接抛异常了。

验证5
核心线程数是3 ,最大线程数是4,队列是SynchronousQueue

  1. ThreadPoolExecutor executor = new ThreadPoolExecutor(3, 4, 5, TimeUnit.SECONDS, new SynchronousQueue<Runnable>());
  1. Exception in thread "main" java.util.concurrent.RejectedExecutionException: Task com.sunlinlin.threaddemo.Main$1@14ae5a5 rejected from java.util.concurrent.ThreadPoolExecutor@7f31245a[Running, pool size = 4, active threads = 4, queued tasks = 0, completed tasks = 0]
  2. at java.util.concurrent.ThreadPoolExecutor$AbortPolicy.rejectedExecution(ThreadPoolExecutor.java:2047)
  3. at java.util.concurrent.ThreadPoolExecutor.reject(ThreadPoolExecutor.java:823)
  4. at java.util.concurrent.ThreadPoolExecutor.execute(ThreadPoolExecutor.java:1369)
  5. at com.sunlinlin.threaddemo.Main.main(Main.java:34)
  6. at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
  7. at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
  8. at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
  9. at java.lang.reflect.Method.invoke(Method.java:498)
  10. at com.intellij.rt.execution.application.AppMain.main(AppMain.java:147)
  11. ---先开三个---
  12. 核心线程数3
  13. 线程池线程数3
  14. 队列任务数0
  15. pool-1-thread-2 run
  16. pool-1-thread-3 run
  17. pool-1-thread-4 run
  18. pool-1-thread-1 run
  1. Exception in thread "main" java.util.concurrent.RejectedExecutionException: Task com.sunlinlin.threaddemo.Main$1@14ae5a5 rejected from java.util.concurrent.ThreadPoolExecutor@7f31245a[Running, pool size = 4, active threads = 4, queued tasks = 0, completed tasks = 0]
  2. at java.util.concurrent.ThreadPoolExecutor$AbortPolicy.rejectedExecution(ThreadPoolExecutor.java:2047)
  3. at java.util.concurrent.ThreadPoolExecutor.reject(ThreadPoolExecutor.java:823)
  4. at java.util.concurrent.ThreadPoolExecutor.execute(ThreadPoolExecutor.java:1369)
  5. at com.sunlinlin.threaddemo.Main.main(Main.java:34)
  6. at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
  7. at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
  8. at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
  9. at java.lang.reflect.Method.invoke(Method.java:498)
  10. at com.intellij.rt.execution.application.AppMain.main(AppMain.java:147)
  11. ---先开三个---
  12. 核心线程数3
  13. 线程池线程数3
  14. 队列任务数0
  15. pool-1-thread-2 run
  16. pool-1-thread-3 run
  17. pool-1-thread-4 run
  18. pool-1-thread-1 run

这次在添加第五个任务时就报错了,因为SynchronousQueue各奔不保存任务,收到一个任务就去创建新线程。所以第五个就会抛异常了。

Java多线程-线程池ThreadPoolExecutor构造方法和规则的更多相关文章

  1. java多线程——线程池源码分析(一)

    本文首发于cdream的个人博客,点击获得更好的阅读体验! 欢迎转载,转载请注明出处. 通常应用多线程技术时,我们并不会直接创建一个线程,因为系统启动一个新线程的成本是比较高的,涉及与操作系统的交互, ...

  2. Java多线程——线程池

    系统启动一个新线程的成本是比较高的,因为它涉及到与操作系统的交互.在这种情况下,使用线程池可以很好的提供性能,尤其是当程序中需要创建大量生存期很短暂的线程时,更应该考虑使用线程池. 与数据库连接池类似 ...

  3. [Java多线程]-线程池的基本使用和部分源码解析(创建,执行原理)

    前面的文章:多线程爬坑之路-学习多线程需要来了解哪些东西?(concurrent并发包的数据结构和线程池,Locks锁,Atomic原子类) 多线程爬坑之路-Thread和Runable源码解析 多线 ...

  4. 跟我学Java多线程——线程池与堵塞队列

    前言 上一篇文章中我们将ThreadPoolExecutor进行了深入的学习和介绍,实际上我们在项目中应用的时候非常少有直接应用ThreadPoolExecutor来创建线程池的.在jdk的api中有 ...

  5. Java - "JUC线程池" ThreadPoolExecutor原理解析

    Java多线程系列--“JUC线程池”02之 线程池原理(一) ThreadPoolExecutor简介 ThreadPoolExecutor是线程池类.对于线程池,可以通俗的将它理解为"存 ...

  6. java多线程----线程池源码分析

    http://www.cnblogs.com/skywang12345/p/3509954.html 线程池示例 在分析线程池之前,先看一个简单的线程池示例. 1 import java.util.c ...

  7. 深入理解Java多线程——线程池

    目录 为什么需要线程池 定义 ThreadPoolExecutor 工作队列workQueue 不同的线程池 Executor 线程池的工作原理 线程池生命周期 线程池增长策略 线程池大小的设置 线程 ...

  8. java多线程--线程池的使用

    程序启动一个新线程的成本是很高的,因为涉及到要和操作系统进行交互,而使用线程池可以很好的提高性能,尤其是程序中当需要创建大量生存期很短的线程时,应该优先考虑使用线程池. 线程池的每一个线程执行完毕后, ...

  9. java多线程-线程池

    线程池(Thread Pool)对于限制应用程序中同一时刻运行的线程数很有用.因为每启动一个新线程都会有相应的性能开销,每个线程都需要给栈分配一些内存等等. 我们可以把并发执行的任务传递给一个线程池, ...

随机推荐

  1. freemark使用总结

    1.下拉框中使用三元表达式: <option value="1类" ${(bean.col5!?string="1类")?string('selected ...

  2. 免费开源ERP-成功案例分析(2)

    Odoo用户案例 Odoo用户概要 关于Odoo全球的用户,我们来看一些数据: Odoo目前全球有300万使用者 Odoo系统上每天新创建的数据库超过1000个 Odoo和Word.Excel.Pow ...

  3. mean项目的分模块开发

    全文字版: 新建maven工程在,作为父工程用于最后集合使用,该工程不需要src,只需要一个pom.xml文件,规定一下依赖版本之类的,再建一个工具类的工程,不需要放配置文件,和工程中方法接口有关的不 ...

  4. cmd实现批量文件的base64加密并双击加密文件后正常运行

    之前,一个朋友让我用cmd做个简单的文档加密(base64),对于不太懂电脑的人来说看不懂就行.但是当那个人点击加密后的文件可以正常运行,问咋写? 其实,像这种要求不高的加密来说,随便下载个加密软件就 ...

  5. Mysql中联合索引的最左匹配原则

    在Mysql建立多列索引(联合索引)有最左前缀的原则,即最左优先. 如果我们建立了一个2列的联合索引(col1,col2),实际上已经建立了两个联合索引(col1).(col1,col2); 如果有一 ...

  6. Linux 文件权限管理

    1.文件权限的概述 在Linux系统下,使用权限来保护资源的安全将是一种不错的选择.系统中每个文件的权限都有可读(r).可写(w)和可执行(x)这三种权限,它们分别对应权限数值4.2 和1.系统为每个 ...

  7. 消息 4900,级别 16,状态 2,第 1 行 对表 'XX.XXX' 执行 ALTER TABLE SWITCH 语句失败。对于已启用更改跟踪的表,不可能切换其分区。请先禁用更改跟踪,再使用 ALTER TABLE SWITCH。

    问题描述: 今天处理切换分区数据的时候出现了这个错误: 消息 4900,级别 16,状态 2,第 1 行 对表 'XX.XXX' 执行 ALTER TABLE SWITCH 语句失败.对于已启用更改跟 ...

  8. LeetCode算法题-Binary Number with Alternating Bits(Java实现)

    这是悦乐书的第292次更新,第310篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第160题(顺位题号是693).给定正整数,检查它是否具有交替位:即它的二进制数的任意两 ...

  9. 【English】20190416

     anti-money laundering反洗钱[ˈænti][ˈlɔːndərɪŋ] misconduct不当行为[ˌmɪsˈkɑːndʌkt]   Currently, she is focus ...

  10. spring mybatics

    spring boot     web.mysql.mybatics.jps 遇到jar包版本不行的直接换个版本Add进去 https://github.com/forezp/SpringBootLe ...