一、四种线程池
Java通过Executors提供四种静态方法来创建线程池
例如:
//创建一个可缓存线程池
ExecutorService cachedThreadPool = Executors.newCachedThreadPool();
//执行任务
cachedThreadPool.execute(Runnable command);
如下:
1.newSingleThreadPool 创建一个单线程化的线程池,它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序(FIFO、LIFO、优先级)执行。
2.newFixedThreadPool 创建一个定长线程池,可控制线程最大并发数,超出的线程会在队列中等待。
3.newScheduledThreadPool 创建一个可定期或者延时执行任务的定长线程池,支持定时及周期性任务执行。
4.newCachedThreadPool 创建一个可缓存线程池,如果线程池长度超过处理需求,可灵活回收空闲线程,若无可回收线程,则新建线程。
二、核心参数
四种线程池本质都是创建ThreadPoolExecutor类,ThreadPoolExecutor构造参数如下
1.int corePoolSize 核心线程数量
2.int maximumPoolSize 最大线程数量
3.long keepAliveTime 超过corePoolSize的线程多久不活动别销毁时间
4.TimeUnit unit 时间单位
5.BlockingQueue workQueue 任务队列
6.ThreadFactory threadFactory 线程池工厂
7.RejectedExecutionHandler handler 拒绝策略
三、线程池工厂 ThreadFactory
所在包位置:java.util.concurrent.ThreadFactory
ThreadFactory接口。这个接口是Java自身提供的,用户可以实现它自定义自己的线程启动方式,可以设置线程名称、类型以及优先级等属性。
ThreadFactory vs Default ThreadFactory:
在一个典型的Java ExecutorService程序中,其线程都需要被指定以何种形式运行,如果程序初始化ExecutorService时没有指定ThreadFactory,程序会采用一个默认的ThreadFactory来生成提交线程,但是对于一个严谨对程序来说,定义自己的ThreadFactory永远是个最佳选择。Why??
1.设置更有描述意义的线程名称。如果使用默认的ThreadFactory,它给线程起名字大概规律就是pool-m-thread-n这个样子,如pool-1-thread-1。但是当你分析一个thread dump时,看着这样的名字就很难知道线程的目的。所以使用一个有描述意义的线程名称是分析追踪问题的clue NO.1。
2.设置线程是否是守护线程,默认的ThreadFactory总是提交非守护线程
3.设置线程优先级,默认ThreadFactory总是提交的一般优先级线程
例子:
CustomThreadFactoryBuilder类实现了一种优雅的Builder Mechanism方式去得到一个自定义ThreadFactory实例。ThreadFactory接口中有一个接受Runnable类型参数的方法newThread(Runnable r),你自己的factory逻辑就应该写在这个方法中,去配置线程名称、优先级、守护线程状态等属性。
public class CustomThreadFactoryBuilder {
private String namePrefix = null;
private boolean daemon = false;
private int priority = Thread.NORM_PRIORITY;
public CustomThreadFactoryBuilder setNamePrefix(String namePrefix) {
if (namePrefix == null) {
throw new NullPointerException();
}
this.namePrefix = namePrefix;
return this;
}
public CustomThreadFactoryBuilder setDaemon(boolean daemon) {
this.daemon = daemon;
return this;
}
public CustomThreadFactoryBuilder setPriority(int priority) {
if (priority < Thread.MIN_PRIORITY){
throw new IllegalArgumentException(String.format(
"Thread priority (%s) must be >= %s", priority, Thread.MIN_PRIORITY));
}
if (priority > Thread.MAX_PRIORITY) {
throw new IllegalArgumentException(String.format(