一、前言

定义
线程池其实就是一种多线程处理形式,处理过程中可以将任务添加到队列中,然后在创建线程后自动启动这些任务。

使用线程池的优势

  • 1、线程和任务分离,提升线程重用性;
  • 2、控制线程并发数量,降低服务器压力,统一管理所有线程;
  • 3、提升系统响应速度,假如创建线程用的时间为T1,执行任务用的时间为T2,销毁线程用的时间为T3,那么使用线程池就免去了T1和T3的时间;

为什么要是用线程池
可以根据系统的需求和硬件环境灵活的控制线程的数量,且可以对所有线程进行统一的管理和控制,从而提高系统的运行效率,降低系统运行运行压力

二、Executor框架

(一)什么是Executor框架?
我们知道线程池就是线程的集合,线程池集中管理线程,以实现线程的重用,降低资源消耗,提高响应速度等。线程用于执行异步任务,单个的线程既是工作单元也是执行机制,从JDK1.5开始,为了把工作单元与执行机制分离开,Executor框架诞生了,他是一个用于统一创建与运行的接口。Executor框架实现的就是线程池的功能。

Executor框架的成员

java 多线程 新框架 java高效多线程框架_java

三、Executor框架结构

1、Executor框架包括3大部分:
(1)任务。也就是工作单元,包括被执行任务需要实现的接口:Runnable接口或者Callable接口;
(2)任务的执行。也就是把任务分派给多个线程的执行机制,包括Executor接口及继承自Executor接口的ExecutorService接口。
(3)异步计算的结果。包括Future接口及实现了Future接口的FutureTask类。

2、Executor框架的使用示意图

java 多线程 新框架 java高效多线程框架_java 多线程 新框架_02

Runnable对象、Callable对象:
使用步骤:

四、创建线程池的方式

(一)通过 ThreadPoolExecutor 构造函数实现

定义MyRunnable类实现Runnable接口,并实现run方法

public class MyRunnable implements Runnable{  
    String name;  
    public MyRunnable(String name){  
        this.name = name;  
    }  
    @Override  
    public void run() {  
        System.out.println(Thread.currentThread().getName() + name + "上山");  
    }  
}

使用ThreadPoolExecutor创建线程池

public class RealPool {  
    //核心线程数  
    static int corePoolSize = 3;  
    //最大线程数  
    static int maximumPoolSize = 6;  
    //超过 corePoolSize 线程数量的线程最大空闲时间  
    static long keepAliveTime = 2;  
    //以秒为时间单位  
    static TimeUnit unit = TimeUnit.SECONDS;  
    //对应创建工作队列,用于存放提交的等待执行任务,此处填写队列大小  
    static int queueSize = 20000;  
  
    public static void main(String[] args) {  
        Long start = System.currentTimeMillis();  
        ThreadPoolExecutor executor = new ThreadPoolExecutor(  
                corePoolSize,  
                maximumPoolSize,  
                keepAliveTime,  
                unit,  
                new ArrayBlockingQueue<>(queueSize),  
                new ThreadPoolExecutor.AbortPolicy()  
        );  
  
        MyRunnable r1 = new MyRunnable("realPool");  
        MyRunnable r2 = new MyRunnable("狗狗");  
        MyRunnable r3 = new MyRunnable("猫猫");  
        executor.execute(r1);  
        executor.execute(r2);  
        executor.execute(r3);  
        executor.shutdown();  
        Long end = System.currentTimeMillis();  
        System.out.println("花费时间为:"+ (end-start) + "ms");  
  
    }  
}
execute与submit

提交任务的类型:

  • execute和submit都属于线程池的方法,execute只能提交Runnable类型的任务
  • submit既能提交Runnable类型任务也能提交Callable类型任务。
    异常:
  • execute会直接抛出任务执行时的异常,可以用try、catch来捕获,和普通线程的处理方式完全一致
  • submit会吃掉异常,可通过Future的get方法将任务执行时的异常重新抛出。
    返回值:
  • execute()没有返回值
  • submit有返回值,所以需要返回值的时候必须使用submit
public void execute(Runnable command) {  
	// 如果实现类是null,则报NullPointerException错误
    if (command == null)  
        throw new NullPointerException();  
     //   主池控制状态ctl是一个原子整数,包装了两个概念字段workerCount,表示有效线程数runState,表示是否正在运行、正在关闭等为了将它们打包成一个int
     // 获取线程的状态
        int c = ctl.get();  
     // 如果当前工作线程少于核心线程,则加一个线程,并把实现类放进去
    if (workerCountOf(c) < corePoolSize) {  
        if (addWorker(command, true))  
            return;  
        c = ctl.get();  
    }  
	 // 如果当前执行的任务数量大于等于 corePoolSize 的时候就会走到这里
	 // 如果线程池正执行,且队列能放新元素,则执行内部
    if (isRunning(c) && workQueue.offer(command)) {  
        int recheck = ctl.get();  
        再次获取线程池状态,如果线程池状态不是 RUNNING 状态就需要从任务队列中移除任务,并尝试判断线程是否全部执行完毕。同时执行拒绝策略。
        if (! isRunning(recheck) && remove(command))  
            reject(command);  
            // 如果当前线程池为空就新创建一个线程并执行。
        else if (workerCountOf(recheck) == 0)  
            addWorker(null, false);  
    }  
    // 创建线程后执行失败,则执行拒绝策略
    else if (!addWorker(command, false))  
        reject(command);  
}

(二)通过工厂类 Executors 来实现

不建议使用
在《阿里巴巴java开发手册》中指出了线程资源必须通过线程池提供,不允许在应用中自行显示的创建线程,这样一方面是线程的创建更加规范,可以合理控制开辟线程的数量;另一方面线程的细节管理交给线程池处理,优化了资源的开销。
而线程池不允许使用Executors去创建,而要通过ThreadPoolExecutor方式,这一方面是由于jdk中Executor框架虽然提供了如newFixedThreadPool()、newSingleThreadExecutor()、newCachedThreadPool()等创建线程池的方法,但都有其局限性,不够灵活;另外由于前面几种方法内部也是通过ThreadPoolExecutor方式实现,使用ThreadPoolExecutor有助于大家明确线程池的运行规则,创建符合自己的业务场景需要的线程池,避免资源耗尽的风险
具体实现