目录
虽然创建和销毁线程比较小(和进程相比),但是当系统中线程数量比较多,这个开销就比较可观的,所以引入线程池
- 目的就是让某些对象被多次重复利用,减少频繁创建和销毁对象带来的开销问题(这些对象一定可以复用的)
- 类比数据库池,创建和销毁数据库的连接就是一个比较耗时的操作,表示当前用户不再使用此连接,就回收到连接池中(同一个连接可以被多个用户使用多次,减少了每次创建和销毁连接的系统开销)
- 同样的,不同的线程也就是run方法的内容不同,线程的大致流程都是一样的,因此为了避免重复创建和销毁线程带来的开销,可以让线程"复用"起来(x线程池最大的好处就是每次减少启动和销毁线程的损耗,提供=高时间和空间的利用率)
- 线程池内部创建好了若干个线程,这些线程都是runnable,只需要从系统中取出任务,就可以立即开始执行
先理清常用的线程池的类和接口的之间的关系
线程池的核心父接口ExecutorService
submit
- 提交一个任务到线程池,线程池就会派遣空闲的线程执行任务
shutdownNow
- 立即尝试终止线程池中所有的线程(是否空闲)
- 停止所有处在空闲状态的线程,正在执行任务的线程,等待任务执行结束再停止,其两方法都是终止所有的线程然后销毁线程池
ThreadPoolExecutor子类的核心构造方法参数
- CallerRunsPolicy 返回给线程的调用者处理
- AbortPolicy 超出负荷的任务直接拒绝,抛出异常
- DiscardOldestPolicy:丢弃队列中最老的任务(排队直接最长的任务)
- DisCardPolicy:丢弃新来的任务
线程池的工作流程
ScheduledExecutorService
Execuctors
使用这个类可以创建JDK内置的四大线程池(Java中带s的类都是工具类,Arrays)
固定大小线程池
动态变化的线程池
单线程的线程池
- 单线程有意义吗,和我们创建一个线程的区别在哪,我们创建一个线程只能执行一个任务,一个任务执行完就销毁了,但是单线程池,虽然同一个时间只能执行一个任务,当这个任务执行结束之后,继续在工作队列调度一个新的任务继续执行(还是减少了销毁和创建线程的开销)
定期线程池
阿里编码归约:尽量不要使用内置线程池,最好根据实际的业务需求,定制线程池自己new ThreadPoolExector对象,传递相关参数