如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了,这样频繁创建线程就会大大降低系统的效率,因为频繁创建线程和销毁线程需要时间。
线程池就是一个容纳多个线程的容器,池中的线程可以反复使用,省去了频繁创建线程对象的操作,节省了大量的时间和资源。
线程池的好处:
- 降低资源消耗
- 提高响应速度
- 提高线程的可管理性
Java中共包含四种线程池:缓存线程池、定长线程池、单线程线程池、周期性任务定长线程池。
缓存线程池:
缓存线程池的长度无限制。
执行流程:
- 判断线程池是否存在空闲线程;
- 存在则使用;不存在则创建并放入线程池再使用
ExecutorService service = Executors.newCachedThreadPool();
//向线程池中加入新的任务Runnable
service.execute(new Runnable() {
@Override
public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
定长线程池:
线程池长度是指定的数值。
执行流程:
- 判断线程池是否存在空闲线程;
- 存在空闲线程则使用;
- 不存在空闲线程且线程池未满时,创建线程并放入线程池进行使用;
- 不存在空闲线程且线程池已满时,等待线程池存在空闲线程。
ExecutorService service = Executors.newFixedThreadPool(2);//固定长度值
//向线程池中加入新的任务Runnable
service.execute(new Runnable() {
@Override
public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
单线程线程池:
和定长线程池中传参1效果一致。
执行流程:
- 判断线程池的线程是否空闲;
- 空闲则使用;不空闲则等待空闲后再使用
ExecutorService service = Executors.newSingleThreadExecutor();
//向线程池中加入新的任务Runnable
service.execute(new Runnable() {
@Override
public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
周期任务 定长线程池:
可以把某个任务定时在某个时期执行,也可以把某个任务周期执行。
执行流程(和定长线程池一致):
- 判断线程池是否存在空闲线程;
- 存在空闲线程则使用;
- 不存在空闲线程且线程池未满时,创建线程并放入线程池进行使用;
- 不存在空闲线程且线程池已满时,等待线程池存在空闲线程。
周期性任务执行时:定时执行,当某个时机触发时,自动执行某任务。
//返回值类型和其余线程池不同
ScheduledExecutorService service = Executors.newScheduledThreadPool(2);
//执行方法也不再是service.execute(任务)
//定时执行一次
//参数:Runnable类型的任务;延迟时长数字;时长的单位——TimeUnit的常量指定
service.schedule(new Runnable() {
@Override
public void run() {
System.out.println("执行一次");
}
},5,TimeUnit.SECONDS);
//周期执行
//参数:1.Runnable类型的任务;
//2.延迟时长数字(第一次执行的延迟);
//3.周期时长数字;
//4.时长的单位——TimeUnit的常量指定
service.scheduleAtFixedRate(new Runnable() {
@Override
public void run() {
System.out.println("执行一次");
}
},5,2,TimeUnit.SECONDS);