Executors
ExecutorService:真正的线程池接口。常见子类ThreadPoolExecutor
void execute(Runnable command) :执行任务/命令,没有返回值,一般用来执行Runnable
Future submit(Callable task):执行任务,有返回值,一般又来执行Callable
void shutdown() :关闭连接池
池技术的好处
- 提高响应速度(减少了创建新线程的时间)
- 降低资源消耗(重复利用线程池中线程,不需要每次都创建)
- 便于线程管理
线程池的7大属性
- corePoolSize 线程池核心线程大小。
线程池中会维护一个最小的线程数量,即使这些线程处理空闲状态,他们也不会自动销毁。 - maximumPoolSize 线程池最大线程数量。
一个任务被提交到线程池以后,首先会找有没有空闲存活线程,如果有则直接执行,如果没有则会缓存到工作队列(后面会介绍)中,如果工作队列满了,才会创建一个新线程,然后从工作队列的头部取出一个任务交由新线程来处理,而将刚提交的任务放入工作队列尾部。线程池不会无限制的去创建新线程,它会有一个最大线程数量的限制,这个数量即由maximunPoolSize指定。 - keepAliveTime 空闲线程存活时间
线程在终止之前可以保持空闲的时间限制。 - unit 空间线程存活时间单位
- workQueue 工作队列
新任务被提交后,会先进入到此工作队列中,任务调度时再从队列中取出任务,没被调度的线程任务会被保持阻塞状态。
队列分为
ArrayBlockingQueue数组队列
LinkedBlockingQuene链表队列
SynchronousQuene不缓存队列,任务进来不会等待调度,如果没有空闲线程,立即在池中创建新的线程,如果线程池达到最大,则执行拒绝策略
PriorityBlockingQueue具有优先级的无界阻塞队列 - threadFactory 线程工厂,创建一个新线程时使用的工厂,可以用来设定线程名、是否为daemon线程等等
- handler 拒绝策略
示例
public class ThreadTest09 {
public static void main(String[] args) {
//创建线程池
ThreadPoolExecutor pool = (ThreadPoolExecutor) Executors.newFixedThreadPool(100);
pool.setCorePoolSize(10);
RunnablePrint runnablePrint = new RunnablePrint();
pool.setCorePoolSize(10);
pool.setKeepAliveTime(1000*60*30, TimeUnit.SECONDS);
//开启池中的2个线程
pool.execute(runnablePrint);
pool.execute(runnablePrint);
//关闭线程池
pool.shutdown();
}
}
class RunnablePrint implements Runnable{
@Override
public void run() {
for (int i = 0; i < 100; i++) {
System.out.println(Thread.currentThread().getName()+"===="+i);
}
}
}