线程的状态
线程池的基本概念
线程是稀缺资源,如果被无限制的创建和销毁,不仅会消耗系统资源,还会降低系统性能能。合理的使用线程池可以更好的利用CPU的空闲时间,对线程进行统一分配,具体有以下几个好处。
1.降低资源消耗
2.提高响应速度
3.提高线程的可管理性
线程池是从JDK1.5之后引入的,线程池中有若干个线程对象,当有任务需要执行时就从池子中拿一个线程过来执行任务,执行完任务之后线程不会被销毁,而是归还到池子给其他要执行的任务复用。
这样就避免了频繁的创建线程和销毁线程浪费系统资源,同时也提高了系统的响应速度。线程池内部有一个任务队列,我们只需要向线程池提交任务,就会自动分配空闲的线程来执行提交的任务。
Executors默认线程池
Java的API中为我们准备好了一些线程池,通过Executors工具类就可以获取线程池。先来看一个默认线程池
//池子的大小默认是0,若无可回收线程,则新建线程,否则就复用已有线程。
ExecutorService executorService = Executors.newCachedThreadPool();
//提交一个线程任务
executorService.submit(() -> {
System.out.println(Thread.currentThread().getName()+"...执行了");
});
//提交另一个线程任务
executorService.submit(() -> {
System.out.println(Thread.currentThread().getName()+"...执行了");
});
//关闭线程池
Executors默认线程池
//创建一个固定长度的线程池。线程的数量最多是3个,不会创建新的线程。
ExecutorService executorService = Executors.newFixedThreadPool(3);
//提交10个线程任务,运行会发现只有3个线程在复用
for (int i = 0; i < 10; i++) {
executorService.submit(() -> {
System.out.println(Thread.currentThread().getName()+"...执行了");
});
}
executorService.shutdown();
ThreadPoolExector创建创建线程池
其实Executors提供的方法创建线程池,底层用的就是ThreadPoolExector来创建的。
我们也可以直接使用ThreadPoolExector创建线程池。下面是ThreadPollExecutor的构造方法
public ThreadPoolExecutor(
int corePoolSize, //核心线程数
int maximumPoolSize, //最大的线程数=核心线程数+临时线程数
long keepAliveTime, //空闲线程最大存活时间
TimeUnit unit, //空闲时间单位
ThreadFactory threadFactory,//创建线程的工厂
RejectedExecutionHandler handler,//任务的拒绝策略
BlockingQueue<Runnable> workQueue //任务队列
)
//创建线程池,线程池最大线程数5个,任务队列最多10个
ThreadPoolExecutor pool =new ThreadPoolExecutor(1,5,2,
TimeUnit.HOURS,
new ArrayBlockingQueue<>(10),
Executors.defaultThreadFactory(),
new ThreadPoolExecutor.AbortPolicy());
//提交创建16个任务,线程池最多可以处理5个、任务队列10个,最多处理15个线程任务,还会多出1个任务
for (int i = 1; i <=16; i++) {
int y=i;
pool.submit(() -> {
System.out.println(Thread.currentThread().getName()+"执行了"+y);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
});
}
任务拒绝策略
ThreadPoolExecutor.AbortPolicy() 丢弃任务并抛出RejectedExecutionException异常,默认策略。
ThreadPoolExecutor.DiscardPolicy() 丢弃任务,但是不抛出异常。不推荐这么做
ThreadPoolExecutor.DiscardOldestPolicy() 抛弃队列中等待最久的任务,然后把当前任务加入队列中
ThreadPoolExecutor.DiscardOldestPolicy() 调用任务的run方法,绕过线程池直接执行。