`
wbj0110
  • 浏览: 1591180 次
  • 性别: Icon_minigender_1
  • 来自: 上海
文章分类
社区版块
存档分类
最新评论

【Java线程】Java线程池ExecutorService

阅读更多

示例

 

  1. import java.util.concurrent.ExecutorService;  
  2. import java.util.concurrent.Executors;  
  3. import java.util.concurrent.ScheduledExecutorService;  
  4.   
  5. public class Ch09_Executor {  
  6.       
  7.    private static void run(ExecutorService threadPool) {  
  8.     for(int i = 1; i < 5; i++) {    
  9.             final int taskID = i;    
  10.             threadPool.execute(new Runnable() {    
  11.                 @Override  
  12.         public void run() {    
  13.                     for(int i = 1; i < 5; i++) {    
  14.                         try {    
  15.                             Thread.sleep(20);// 为了测试出效果,让每次任务执行都需要一定时间    
  16.                         } catch (InterruptedException e) {    
  17.                             e.printStackTrace();    
  18.                         }    
  19.                         System.out.println("第" + taskID + "次任务的第" + i + "次执行");    
  20.                     }    
  21.                 }    
  22.             });    
  23.         }    
  24.         threadPool.shutdown();// 任务执行完毕,关闭线程池    
  25.    }  
  26.       
  27.     public static void main(String[] args) {  
  28.         // 创建可以容纳3个线程的线程池  
  29.         ExecutorService fixedThreadPool = Executors.newFixedThreadPool(3);  
  30.           
  31.         // 线程池的大小会根据执行的任务数动态分配  
  32.         ExecutorService cachedThreadPool = Executors.newCachedThreadPool();  
  33.           
  34.             // 创建单个线程的线程池,如果当前线程在执行任务时突然中断,则会创建一个新的线程替代它继续执行任务    
  35.         ExecutorService singleThreadPool = Executors.newSingleThreadExecutor();  
  36.           
  37.         // 效果类似于Timer定时器  
  38.         ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(3);  
  39.           
  40.         run(fixedThreadPool);  
  41. //      run(cachedThreadPool);  
  42. //      run(singleThreadPool);  
  43. //      run(scheduledThreadPool);  
  44.     }  
  45.   
  46. }  

 

 

CachedThreadPool

 

CachedThreadPool会创建一个缓存区,将初始化的线程缓存起来。会终止并且从缓存中移除已有60秒未被使用的线程。

如果线程有可用的,就使用之前创建好的线程,

如果线程没有可用的,就新创建线程。

 

  • 重用:缓存型池子,先查看池中有没有以前建立的线程,如果有,就reuse;如果没有,就建一个新的线程加入池中
  • 使用场景:缓存型池子通常用于执行一些生存期很短的异步型任务,因此在一些面向连接的daemon型SERVER中用得不多。
  • 超时:能reuse的线程,必须是timeout IDLE内的池中线程,缺省timeout是60s,超过这个IDLE时长,线程实例将被终止及移出池。
  • 结束:注意,放入CachedThreadPool的线程不必担心其结束,超过TIMEOUT不活动,其会自动被终止。

 

  1. // 线程池的大小会根据执行的任务数动态分配  
  2. ExecutorService cachedThreadPool = Executors.newCachedThreadPool();  
  3.   
  4. public static ExecutorService newCachedThreadPool() {  
  5.     return new ThreadPoolExecutor(0,                 //core pool size  
  6.                                   Integer.MAX_VALUE, //maximum pool size  
  7.                                   60L,               //keep alive time  
  8.                                   TimeUnit.SECONDS,  
  9.                                   new SynchronousQueue<Runnable>());  
  10. }  


执行结果:

 

 

  1. 第1次任务的第1次执行  
  2. 第4次任务的第1次执行  
  3. 第3次任务的第1次执行  
  4. 第2次任务的第1次执行  
  5. 第3次任务的第2次执行  
  6. 第4次任务的第2次执行  
  7. 第2次任务的第2次执行  
  8. 第1次任务的第2次执行  
  9. 第2次任务的第3次执行  
  10. 第4次任务的第3次执行  
  11. 第3次任务的第3次执行  
  12. 第1次任务的第3次执行  
  13. 第2次任务的第4次执行  
  14. 第1次任务的第4次执行  
  15. 第3次任务的第4次执行  
  16. 第4次任务的第4次执行  


4个任务是交替执行的

 

 

FixedThreadPool

 

在FixedThreadPool中,有一个固定大小的池。

如果当前需要执行的任务超过池大小,那么多出的任务处于等待状态,直到有空闲下来的线程执行任务,

如果当前需要执行的任务小于池大小,空闲的线程也不会去销毁。

 

  • 重用:fixedThreadPool与cacheThreadPool差不多,也是能reuse就用,但不能随时建新的线程
  • 固定数目:其独特之处在于,任意时间点,最多只能有固定数目的活动线程存在,此时如果有新的线程要建立,只能放在另外的队列中等待,直到当前的线程中某个线程终止直接被移出池子
  • 超时:和cacheThreadPool不同,FixedThreadPool没有IDLE机制(可能也有,但既然文档没提,肯定非常长,类似依赖上层的TCP或UDP IDLE机制之类的),
  • 使用场景:所以FixedThreadPool多数针对一些很稳定很固定的正规并发线程,多用于服务器
  • 源码分析:从方法的源代码看,cache池和fixed 池调用的是同一个底层池,只不过参数不同:
    fixed池线程数固定,并且是0秒IDLE(无IDLE)
    cache池线程数支持0-Integer.MAX_VALUE(显然完全没考虑主机的资源承受能力),60秒IDLE

 

  1. // 创建可以容纳3个线程的线程池  
  2. ExecutorService fixedThreadPool = Executors.newFixedThreadPool(3);  
  3.   
  4. public static ExecutorService newFixedThreadPool(int nThreads) {  
  5.         return new ThreadPoolExecutor(nThreads, //core pool size  
  6.                                       nThreads, //maximum pool size  
  7.                                       0L,       //keep alive time  
  8.                                       TimeUnit.MILLISECONDS,  
  9.                                       new LinkedBlockingQueue<Runnable>());  
  10. }  

 

 

执行结果:

  1. 第1次任务的第1次执行  
  2. 第3次任务的第1次执行  
  3. 第2次任务的第1次执行  
  4. 第3次任务的第2次执行  
  5. 第2次任务的第2次执行  
  6. 第1次任务的第2次执行  
  7. 第3次任务的第3次执行  
  8. 第1次任务的第3次执行  
  9. 第2次任务的第3次执行  
  10. 第3次任务的第4次执行  
  11. 第1次任务的第4次执行  
  12. 第2次任务的第4次执行  
  13. 第4次任务的第1次执行  
  14. 第4次任务的第2次执行  
  15. 第4次任务的第3次执行  
  16. 第4次任务的第4次执行  


创建了一个固定大小的线程池,容量为3,然后循环执行了4个任务。由输出结果可以看到,前3个任务首先执行完,然后空闲下来的线程去执行第4个任务

 

SingleThreadExecutor 

SingleThreadExecutor得到的是一个单个的线程,这个线程会保证你的任务执行完成。

如果当前线程意外终止,会创建一个新线程继续执行任务,这和我们直接创建线程不同,也和newFixedThreadPool(1)不同。

 

  1. // 创建单个线程的线程池,如果当前线程在执行任务时突然中断,则会创建一个新的线程替代它继续执行任务    
  2. ExecutorService singleThreadPool = Executors.newSingleThreadExecutor();  
  3.   
  4. public static ExecutorService newSingleThreadExecutor() {  
  5.         return new FinalizableDelegatedExecutorService  
  6.             (new ThreadPoolExecutor(1,  //core pool size  
  7.                                     1,  //maximum pool size  
  8.                                     0L, //keep alive time  
  9.                                     TimeUnit.MILLISECONDS,  
  10.                                     new LinkedBlockingQueue<Runnable>()));  
  11. }  

 

 

执行结果:

 

  1. 第1次任务的第1次执行  
  2. 第1次任务的第2次执行  
  3. 第1次任务的第3次执行  
  4. 第1次任务的第4次执行  
  5. 第2次任务的第1次执行  
  6. 第2次任务的第2次执行  
  7. 第2次任务的第3次执行  
  8. 第2次任务的第4次执行  
  9. 第3次任务的第1次执行  
  10. 第3次任务的第2次执行  
  11. 第3次任务的第3次执行  
  12. 第3次任务的第4次执行  
  13. 第4次任务的第1次执行  
  14. 第4次任务的第2次执行  
  15. 第4次任务的第3次执行  
  16. 第4次任务的第4次执行  


4个任务是顺序执行的



 

ScheduledThreadPool

ScheduledThreadPool是一个固定大小的线程池,与FixedThreadPool类似,执行的任务是定时执行

 

  1. // 效果类似于Timer定时器  
  2. ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(3);  
  3.   
  4. public ScheduledThreadPoolExecutor(int corePoolSize) {  
  5.         super(corePoolSize,      //core pool size  
  6.               Integer.MAX_VALUE, //maximum pool size  
  7.               0,                 //keep alive time  
  8.               TimeUnit.NANOSECONDS,  
  9.               new DelayedWorkQueue());  
  10. }  


执行结果:

 

 

  1. 第1次任务的第1次执行  
  2. 第2次任务的第1次执行  
  3. 第3次任务的第1次执行  
  4. 第2次任务的第2次执行  
  5. 第1次任务的第2次执行  
  6. 第3次任务的第2次执行  
  7. 第2次任务的第3次执行  
  8. 第1次任务的第3次执行  
  9. 第3次任务的第3次执行  
  10. 第2次任务的第4次执行  
  11. 第1次任务的第4次执行  
  12. 第3次任务的第4次执行  
  13. 第4次任务的第1次执行  
  14. 第4次任务的第2次执行  
  15. 第4次任务的第3次执行  
  16. 第4次任务的第4次执行  


——与FixedThreadPool的区别?

分享到:
评论

相关推荐

    在spring boot中使用java线程池ExecutorService的讲解

    在 Spring Boot 中使用 Java 线程池 ExecutorService 的讲解 Spring Boot 作为一个流行的 Java 框架,提供了许多便捷的功能来帮助开发者快速构建应用程序。其中之一就是使用 Java 线程池 ExecutorService 来管理...

    java多线程,对多线程,线程池进行封装,方便使用

    Java的ExecutorService和ThreadPoolExecutor是线程池的核心组件。 1. **线程创建和销毁成本** 创建和销毁线程是有一定开销的,包括虚拟内存分配、上下文切换等。线程池通过重用已存在的线程,避免了频繁创建和销毁...

    18.【线程池、Lambda表达式】_java线程_lambda线程池_meantbs3_

    首先,让我们深入理解Java线程。在多核处理器时代,利用多线程能有效利用系统资源,提高程序并发执行的能力。Java提供了`Thread`类来创建线程,但直接使用`Thread`会带来资源开销,如频繁创建和销毁线程。为了解决这...

    java_thread_cn.rar_Java 线程池_java thread books_java线程_线程池_线程池调度

    Java线程调度策略包括抢占式调度和合作式调度。在Java中,默认采用的是抢占式调度,即线程的执行优先级由JVM决定,高优先级的线程可能会中断低优先级线程的执行。然而,开发者可以通过设置线程的优先级来影响调度,...

    java线程、线程池、xml解析入门

    Java线程、线程池和XML解析是Java编程中至关重要的三个概念,它们在实际开发中扮演着不可或缺的角色。下面将分别对这三个主题进行深入的介绍。 首先,我们来看Java线程。线程是程序执行的最小单位,一个进程可以...

    Java中多线程的使用线程池.docx

    Java中的线程池是多线程编程中一种高效、可管理的执行机制。它通过预先创建并维护一组线程,避免了频繁地创建和销毁线程带来的开销,从而提高了程序的性能和响应速度。线程池的核心概念包括以下几个方面: 1. **...

    java 线程池源代码

    Java线程池是一种高效管理线程的机制,它允许开发者预先创建一定数量的线程,然后根据需求将任务提交到线程池中进行执行。线程池的核心在于它能够有效地控制运行的线程数量,避免因为频繁创建和销毁线程而产生的性能...

    JAVA使用线程池查询大批量数据

    除了`ThreadPoolExecutor`,Java还提供了`Executors`工具类,它提供了一些预设的线程池配置,如`newFixedThreadPool`(固定大小线程池)、`newSingleThreadExecutor`(单线程线程池)等,方便开发者快速创建线程池。...

    26_多线程_第1天(Thread、线程创建、线程池)_讲义

    为了解决这个问题,Java提供了ExecutorService和ThreadPoolExecutor,它们是线程池的概念。线程池可以预先配置好一定数量的线程,当有新的任务提交时,线程池会复用已存在的线程,而不是每次都创建新的。这样可以...

    多线程控制,线程池模式,HTTP线程

    Java中的ExecutorService接口和ThreadPoolExecutor类是实现线程池的经典工具。 HTTP线程则是在网络编程中,处理HTTP协议请求和响应的线程。在服务器端,每个HTTP请求通常由一个单独的线程来处理,以便并发处理多个...

    Java/Android线程池演示Demo

    Java的`java.util.concurrent`包提供了`ExecutorService`接口,它是线程池的主要入口。通过实现这个接口,我们可以控制线程的执行方式,如设置最大线程数、处理任务队列等。`ExecutorService`通常与`...

    java线程和线程池的使用.docx

    下面将详细介绍 Java 中线程的创建方式以及线程池的使用。 1. 创建线程的方式: Java 提供了两种主要的创建线程的方式: - **实现 Runnable 接口**:你的类实现 Runnable 接口,并重写 `run()` 方法。然后通过...

    java多线程查询数据库

    Java中的线程池是通过`java.util.concurrent.ExecutorService`接口和其实现类(如`ThreadPoolExecutor`)来创建的。线程池可以有效地管理线程资源,避免频繁创建和销毁线程带来的开销。通过设置线程池的参数(核心...

    JAVA集中常用的线程池比较.pdf

    Java线程池是一种高效管理线程的工具,它允许开发者预先配置一组线程,以便在处理并发任务时能更好地控制系统的资源。线程池的概念源于服务器应用程序中对大量短小任务处理的需求,避免频繁创建和销毁线程带来的性能...

    单线程 多线程 线程池 计算随机数的一价和

    例如,Java的`Thread`类代表线程,`ExecutorService`接口和`ThreadPoolExecutor`类用于实现线程池,而Python的`threading`模块提供了线程相关的功能,`concurrent.futures`模块则提供了线程池的实现。理解并熟练掌握...

    轻量级java多线程池demo

    Java内置的`java.util.concurrent`包提供了一个强大的线程池实现——`ExecutorService`,通过`ThreadPoolExecutor`类可以自定义线程池配置。线程池通常包含核心线程数、最大线程数、线程空闲时间、工作队列等参数,...

    多线程控制、线程池模型、HTTP线程.rar

    Java中的ExecutorService接口和ThreadPoolExecutor类提供了线程池的实现。线程池参数包括核心线程数、最大线程数、线程存活时间、任务队列等,合理配置这些参数对系统性能有很大影响。 3. **HTTP线程**: HTTP(超...

    java多种类型的线程池操作

    - **ExecutorService接口**:它是Java线程池的核心接口,提供了提交任务、关闭线程池和管理线程池状态的方法。 - **ThreadPoolExecutor类**:实现了ExecutorService接口,提供了更多自定义线程池的参数,如核心...

    java服务器端Socket线程池

    上述代码展示了如何实现一个简单的Java线程池,用于处理Socket连接。线程池的核心概念包括以下几个部分: 1. **线程池容量**:由`maxThreads`、`minSpareThreads`和`maxSpareThreads`三个参数控制。`maxThreads`是...

    JAVA 写的SOCKET线程池

    Java中的Socket线程池是一种高效的网络编程模型,它结合了Socket通信和多线程技术,以提高服务端处理客户端请求的并发性能。对于初学者来说,理解并掌握这个概念至关重要,因为这能帮助他们构建更稳定、可扩展的网络...

Global site tag (gtag.js) - Google Analytics