zoukankan      html  css  js  c++  java
  • JAVA线程池

    在这里插入图片描述
    submit()和execute()都是 ExecutorService 的方法,是添加线程到线程池中。 submit()有返回值,返回future。 execute()没有返回值。

    1、线程池的优势
    (1)、降低系统资源消耗,通过重用已存在的线程,降低线程创建和销毁造成的消耗;
    (2)、提高系统响应速度,当有任务到达时,通过复用已存在的线程,无需等待新线程的创建便能立即执行;
    (3)方便线程并发数的管控。因为线程若是无限制的创建,可能会导致内存占用过多而产生OOM,并且会造成cpu过度切换(cpu切换线程是有时间成本的(需要保持当前执行线程的现场,并恢复要执行线程的现场))。
    (4)提供更强大的功能,延时定时线程池。

    2、线程池的主要参数
    构造方法:

    //构造方法
    public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit, BlockingQueue<Runnable> workQueue);
      
    public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit,BlockingQueue<Runnable> workQueue,ThreadFactory threadFactory);
      
    public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit,BlockingQueue<Runnable> workQueue,RejectedExecutionHandler handler);
      
    public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit,BlockingQueue<Runnable> workQueue,ThreadFactory threadFactory,RejectedExecutionHandler handler)
    

    参数分析:
    1、corePoolSize(线程池基本大小、核心线程数量):当向线程池提交一个任务时,若线程池已创建的线程数小于corePoolSize,即便此时存在空闲线程,也会通过创建一个新线程来执行该任务,直到已创建的线程数大于或等于corePoolSize时,(除了利用提交新任务来创建和启动线程(按需构造),也可以通过 prestartCoreThread() 或 prestartAllCoreThreads() 方法来提前启动线程池中的基本线程。)

    2、maximumPoolSize(线程池最大大小):线程池所允许的最大线程个数。当队列满了,且已创建的线程数小于maximumPoolSize,则线程池会创建新的线程来执行任务。另外,对于无界队列,可忽略该参数。

    3、keepAliveTime(线程存活保持时间)当线程池中线程数大于核心线程数时,线程的空闲时间如果超过线程存活时间,那么这个线程就会被销毁,直到线程池中的线程数小于等于核心线程数。

    4、workQueue(任务队列):用于传输和保存等待执行任务的阻塞队列。

    5、threadFactory(线程工厂):用于创建新线程。threadFactory创建的线程也是采用new Thread()方式,threadFactory创建的线程名都具有统一的风格:pool-m-thread-n(m为线程池的编号,n为线程池内的线程编号)。

    5、handler(线程饱和策略):当线程池和队列都满了,再加入线程会执行此策略。

    
    ThreadPoolExecutor.AbortPolicy         : 丢弃任务并抛出RejectedExecutionException异常。 
    ThreadPoolExecutor.DiscardPolicy       :丢弃任务但不抛出异常。 
    ThreadPoolExecutor.DiscardOldestPolicy :丢弃队列最前面的任务,执行新提交的任务
    ThreadPoolExecutor.CallerRunsPolicy    :调用线程处理该任务
    

    3、线程池流程
    在这里插入图片描述
    1、判断核心线程池是否已满,没满则创建一个新的工作线程来执行任务。已满则。
    2、判断任务队列是否已满,没满则将新提交的任务添加在工作队列,已满则。
    3、判断整个线程池是否已满,没满则创建一个新的工作线程来执行任务,已满则执行饱和策略。

    (1、判断线程池中当前线程数是否大于核心线程数,如果小于,在创建一个新的线程来执行任务,如果大于则
    2、判断任务队列是否已满,没满则将新提交的任务添加在工作队列,已满则。
    3、判断线程池中当前线程数是否大于最大线程数,如果小于,则创建一个新的线程来执行任务,如果大于,则执行饱和策略。)

    4、线程池为什么需要使用(阻塞)队列?
    回到了非线程池缺点中的第3点:
    1、因为线程若是无限制的创建,可能会导致内存占用过多而产生OOM ( Out Of Memory 内存用完) ,并且会造成cpu过度切换。

    另外回到了非线程池缺点中的第1点:
    2、创建线程池的消耗较高。
    或者下面这个网上并不高明的回答:
    2、线程池创建线程需要获取mainlock这个全局锁,影响并发效率,阻塞队列可以很好的缓冲。

    5、如何配置线程池
    CPU密集型任务
    尽量使用较小的线程池,一般为CPU核心数+1。 因为CPU密集型任务使得CPU使用率很高,若开过多的线程数,会造成CPU过度切换。

    IO密集型任务
    可以使用稍大的线程池,一般为2*CPU核心数。 IO密集型任务CPU使用率并不高,因此可以让CPU在等待IO的时候有其他线程去处理别的任务,充分利用CPU时间。

    混合型任务
    可以将任务分成IO密集型和CPU密集型任务,然后分别用不同的线程池去处理。 只要分完之后两个任务的执行时间相差不大,那么就会比串行执行来的高效。
    因为如果划分之后两个任务执行时间有数据级的差距,那么拆分没有意义。
    因为先执行完的任务就要等后执行完的任务,最终的时间仍然取决于后执行完的任务,而且还要加上任务拆分与合并的开销,得不偿失。

    6、java中提供的线程池 (四种线程池的静态创建方法如下: )
    Executors类提供了4种不同的线程池:newCachedThreadPool, newFixedThreadPool, newScheduledThreadPool, newSingleThreadExecutor
    (1) newCachedThreadPool
    创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程。示例代码如下:

    package test;  
    import java.util.concurrent.ExecutorService;  
    import java.util.concurrent.Executors;  
    public class ThreadPoolExecutorTest {  
     public static void main(String[] args) {  
      ExecutorService cachedThreadPool = Executors.newCachedThreadPool();  
      for (int i = 0; i < 10; i++) {  
       final int index = i;  
       try {  
        Thread.sleep(index * 1000);  
       } catch (InterruptedException e) {  
        e.printStackTrace();  
       }  
       cachedThreadPool.execute(new Runnable() {  
        public void run() {  
         System.out.println(index);  
        }  
       });  
      }  
     }  
    }  
    

    (2) newFixedThreadPool

    创建一个定长线程池,可控制线程最大并发数,超出的线程会在队列中等待。示例代码如下:

    package test;  
    import java.util.concurrent.ExecutorService;  
    import java.util.concurrent.Executors;  
    public class ThreadPoolExecutorTest {  
     public static void main(String[] args) {  
      ExecutorService fixedThreadPool = Executors.newFixedThreadPool(3);  
      for (int i = 0; i < 10; i++) {  
       final int index = i;  
       fixedThreadPool.execute(new Runnable() {  
        public void run() {  
         try {  
          System.out.println(index);  
          Thread.sleep(2000);  
         } catch (InterruptedException e) {  
          e.printStackTrace();  
         }  
        }  
       });  
      }  
     }  
    }  
    

    因为线程池大小为3,每个任务输出index后sleep 2秒,所以每两秒打印3个数字。
    定长线程池的大小最好根据系统资源进行设置。如Runtime.getRuntime().availableProcessors()

    (3) newScheduledThreadPool

    创建一个定长线程池,支持定时及周期性任务执行。延迟执行示例代码如下:

    package test;  
    import java.util.concurrent.Executors;  
    import java.util.concurrent.ScheduledExecutorService;  
    import java.util.concurrent.TimeUnit;  
    public class ThreadPoolExecutorTest {  
     public static void main(String[] args) {  
      ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(5);  
      scheduledThreadPool.schedule(new Runnable() {  
       public void run() {  
        System.out.println("delay 3 seconds");  
       }  
      }, 3, TimeUnit.SECONDS);  
     }  
    }  
    

    表示延迟3秒执行。

    定期执行示例代码如下:

    package test;  
    import java.util.concurrent.Executors;  
    import java.util.concurrent.ScheduledExecutorService;  
    import java.util.concurrent.TimeUnit;  
    public class ThreadPoolExecutorTest {  
     public static void main(String[] args) {  
      ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(5);  
      scheduledThreadPool.scheduleAtFixedRate(new Runnable() {  
       public void run() {  
        System.out.println("delay 1 seconds, and excute every 3 seconds");  
       }  
      }, 1, 3, TimeUnit.SECONDS);  
     }  
    }  
    

    表示延迟1秒后每3秒执行一次。

    (4) newSingleThreadExecutor

    创建一个单线程化的线程池,它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序(FIFO, LIFO, 优先级)执行。示例代码如下:

    package test;  
    import java.util.concurrent.ExecutorService;  
    import java.util.concurrent.Executors;  
    public class ThreadPoolExecutorTest {  
     public static void main(String[] args) {  
      ExecutorService singleThreadExecutor = Executors.newSingleThreadExecutor();  
      for (int i = 0; i < 10; i++) {  
       final int index = i;  
       singleThreadExecutor.execute(new Runnable() {  
        public void run() {  
         try {  
          System.out.println(index);  
          Thread.sleep(2000);  
         } catch (InterruptedException e) {  
          e.printStackTrace();  
         }  
        }  
       });  
      }  
     }  
    }  
    
  • 相关阅读:
    eclipse注释模板设置(未整理)
    10大最适合编程的字体推荐下载,让代码看起来更美更舒服!
    系统里有Courier New字体 Eclipse没有这个字体选项
    Hadoop安装教程_集群/分布式配置
    Hadoop安装教程_单机/伪分布式配置
    VMware中三种网络连接的区别
    修改Tomcat的默认访问目录
    Ubuntu 16.04服务器 配置
    Ubuntu 16.04服务器 软件的安装及配置
    排序算法 -- 桶排序
  • 原文地址:https://www.cnblogs.com/mhlsky/p/13848868.html
Copyright © 2011-2022 走看看