Java多线程与高并发七本来想写ThreadLocal源码的,实在是看底层代码看不动了!先搁置吧。Java多线程与高并发八给同学们分享下面试中常考的线程池的七大参数!

为什么要用线程池

       防止频繁创建和销毁线程,让每个线程可以多次使用,防止消耗过多内存,所以我们使用线程池。

为什么不用JDK自带线程池

        像下面这样定义线程池,不香吗?为什么不用呢?

ExecutorService executorService = Executors.newFixedThreadPool(5);

        阿里的《阿里巴巴 Java 开发手册》已经说得很清楚明白了,我这里就直接盗用吧。

java 多线程池工具 java多线程并发编程 线程池_线程池

自己创建线程池的目的是让我们自己更加明确线程池的运行规则!

如何创建线程池

  • 线程池的七大参数
public ThreadPoolExecutor(int corePoolSize,
                              int maximumPoolSize,
                              long keepAliveTime,
                              TimeUnit unit,
                              BlockingQueue<Runnable> workQueue,
                              ThreadFactory threadFactory,
                              RejectedExecutionHandler handler) {
        //实现略
    }
corePoolSize:线程池持有的可用的线程数量
maximumPoolSize:当队列workQueue满了过后,线程池可用的最大线程数量
keepAliveTime:大于核心线程数的帮工线程们等待新任务到来的等待时间,超过该时间,帮工线程们还没有可执行的任务,这些线程就会关闭
unit:上个参数keepAliveTime的单位,通常是以,TimeUnit.SECONDS,这样给出的,表示秒
workQueue:任务堆积的等待队列,比如当前核心线程数是5,来了10个,那么后来的5个就会放到该队列中
threadFactory:线程工厂,主要是给线程赋予特定名称
handler:拒绝策略,当队列满了,帮工线程们也起来了,还在往线程池里加线程该怎么处理?这个策略就是定义怎么处理的。
  • 线程池的创建案例
import java.util.concurrent.*;

/**
 * 线程池测试
 *
 * @author zab
 * @date 2020-01-21 22:40
 */
public class ThreadPoolTest {
    private static final int CORE_POOL_SIZE = 5;
    private static final int MAX_POOL_SIZE = 8;
    private static final int KEEP_ALIVE_TIME = 60;

    public static void main(String[] args) throws Exception{

        ThreadPoolTest t = new ThreadPoolTest();

        ThreadPoolExecutor e = t.getThreadPool2();

        for (int i = 0; i < 100; i++) {
            e.execute(t.new MyRunnable(i));
        }
        
    }

    private ThreadPoolExecutor getThreadPool0() {
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(
                CORE_POOL_SIZE,
                MAX_POOL_SIZE,
                KEEP_ALIVE_TIME,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(10),
                new MyThreadFactory("whatBusiness"),
                new ThreadPoolExecutor.AbortPolicy()
        );

        return threadPoolExecutor;
    }

    private ThreadPoolExecutor getThreadPool1() {
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(
                CORE_POOL_SIZE,
                MAX_POOL_SIZE,
                KEEP_ALIVE_TIME,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(10),
                new MyThreadFactory("whatBusiness"),
                new ThreadPoolExecutor.DiscardPolicy()
        );

        return threadPoolExecutor;
    }

    private ThreadPoolExecutor getThreadPool2() {
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(
                CORE_POOL_SIZE,
                MAX_POOL_SIZE,
                KEEP_ALIVE_TIME,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(10),
                new MyThreadFactory("whatBusiness"),
                new ThreadPoolExecutor.DiscardOldestPolicy()
        );

        return threadPoolExecutor;
    }

    private ThreadPoolExecutor getThreadPool3() {
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(
                CORE_POOL_SIZE,
                MAX_POOL_SIZE,
                KEEP_ALIVE_TIME,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(10),
                new MyThreadFactory("whatBusiness"),
                new ThreadPoolExecutor.CallerRunsPolicy()
        );

        return threadPoolExecutor;
    }

    private ThreadPoolExecutor getThreadPool4() {
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(
                CORE_POOL_SIZE,
                MAX_POOL_SIZE,
                KEEP_ALIVE_TIME,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(10),
                new MyThreadFactory("whatBusiness"),
                new MyRejectedExecutionHandler()
        );

        return threadPoolExecutor;
    }

    class MyRejectedExecutionHandler implements RejectedExecutionHandler {

        @Override
        public void rejectedExecution(Runnable r, ThreadPoolExecutor executor) {
            System.out.println("做自己想要的逻辑");
            System.out.println("核心池大小:" + executor.getCorePoolSize());
        }
    }


    class MyThreadFactory implements ThreadFactory {

        private String threadName;

        public MyThreadFactory(String threadName) {
            this.threadName = threadName;
        }

        @Override
        public Thread newThread(Runnable r) {
            Thread t = new Thread(r);
            t.setName(threadName+t.getName());
            return t;
        }
    }

    class MyRunnable implements Runnable {

        int count = 0;

        public MyRunnable(int count){
            this.count = count;
        }

        @Override
        public void run() {
            try {
                Thread.sleep(2000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("Thread---"+Thread.currentThread().getName()+"----count:"+count);
        }
    }
}

这里定义getThreadPool的几个方法主要是获取不同拒绝策略的线程池:

getThreadPool0:获取拒绝策略的线程池
getThreadPool1:获取丢弃策略的线程池
getThreadPool2:获取丢弃最老策略的线程池
getThreadPool3:获取调用者执行策略的线程池
getThreadPool4:获取自定义策略的线程池

然后自定义了几个内部类:

MyRejectedExecutionHandler自定义拒绝策略
MyThreadFactory自定义线程工厂
MyRunnable自定义线程类

main方法很简单,就是获取不同的线程池,for循环跑100次,然后撑爆线程池的等待队列,测试不同拒绝策略的表现。

线程池拒绝策略分析

1、抛异常策略

java 多线程池工具 java多线程并发编程 线程池_等待队列_02

可以看到核心线程数5,最大线程数8,循环执行100次,当核心线程数满了后,再来新的线程,就会往等待队列里放,这里的等待队列是一个ArrayBlockingQueue,容量是10,再来线程满了后,会创建【帮工线程】,可以看到线程名最大是7,说明该测试已经把【帮工线程】都用上了,再来线程就会执行拒绝策略,可以看到控制台有抛异常。

2、丢弃策略

java 多线程池工具 java多线程并发编程 线程池_自定义_03

丢弃策略什么都不干,其他分析同拒绝策略,值得注意的是,控制台5、6、7线程对应的计数为什么是15、16、17。设想一个场景:

银行有八个窗口,其中五个是打开状态,有人络绎不绝的来办理业务,一个、两个、三个...当第五个人来的时候,如果前面的人都未办理完业务,那么第五个人一定走向最后一个打开的窗口,这时候又来了第六个、第七个...这个时候,保安见状,略有所思:今儿个业务太好了,得让客户排队了!最后第十五个人来的时候,前面都还没人办理完业务,排队都排了10个人了。。。第十六个人来的时候,保安终于坐不住了,去通知大堂经理,大堂经理见状,立即安排了三个业务员把剩余的窗口打开,保安就安排第十六、十七、十八个客户去了新开的窗口办理业务了,而排队的10个人依旧排着队。

以上场景看似不合理,但是确实是代码执行中看到的现象的形象描述。

3、丢弃最老等待线程策略

java 多线程池工具 java多线程并发编程 线程池_等待队列_04

 这个可以看到,等待最久的线程都被忽略了,新来的得到执行。控制台对应的count是for循环的下标,最后执行的下标都达到99了,这说明循环中比较小的数,例如第77次、第88次这些都是等待比较久的线程,都已经被丢弃了。

结合上述案例来说,就好比先来的等待办理业务的客户等不及了,离开银行了。

4、谁调用谁执行策略

java 多线程池工具 java多线程并发编程 线程池_等待队列_05

这个比较简单,就是main方法调用的线程池执行的,那么线程池的线程跑不完的就给调用者main方法执行。

 5、自定义策略

java 多线程池工具 java多线程并发编程 线程池_线程池_06

这个可以看到,一旦有新的线程既排不了队,又不能执行,就会执行自定义的策略。这里没有策略,只是输出信息。

线程池新线程添加流程

  1. 当前池内线程数小于核心线程数时,有任务来,新建一个线程
  2. 当前池内线程数大于核心线程数时,有任务来,队列没排满时,就排队
  3. 当前池内线程数大于核心线程数时,有任务来,队列排满时,线程池继续new线程处理任务,直到线程池线程数等于最大线程池数为止
  4. 当前池内线程数等于最大线程数时,有任务来,队列未满时,排队处理;队列满时,执行拒绝策略