Java线程池
一、什么是线程池
简单来说,线程池就是一个容纳多个线程的容器,池中的线程可以反复使用。
二、为什么要用线程池
如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了,由于频繁创建线程和销毁线程需要时间,所以频繁创建线程就会大大降低系统的效率。而池中的线程可以反复使用,省去了频繁创建线程对象的操作,降低资源消耗,提高响应速度,提高线程的可管理性。
三、在哪里用线程池
从第二点我们可以很清晰的看出,如果我们需要反复使用多个线程的时候,就可以使用线程池。
四、如何使用线程池
首先我们要知道,线程池大致分为四个:缓存线程池、定长线程池、单线程线程池、周期性任务定长线程池,下面我们分别介绍他们的用法。
-
缓存线程池(长度无限制)
执行流程:- 判断线程池是否存在空闲线程
- 存在则使用
- 不存在,则创建线程 并放入线程池, 然后使用
代码如下:
ExecutorService service = Executors.newCachedThreadPool();
//向线程池中 加入 新的任务
service.execute(new Runnable() {
@Override
public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
service.execute(new Runnable() {
@Override
public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
service.execute(new Runnable() {
@Override
public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
-
定长线程池(长度为指定数值)
执行流程:- 判断线程池是否存在空闲线程
- 存在则使用
- 不存在空闲线程,且线程池未满的情况下,则创建线程 并放入线程池, 然后使用
- 不存在空闲线程,且线程池已满的情况下,则等待线程池存在空闲线程
代码如下:
ExecutorService service = Executors.newFixedThreadPool(2);
service.execute(new Runnable() {
@Override
public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
service.execute(new Runnable() {
@Override
public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
-
单线程线程池(效果与定长线程池创建时传入数值1 效果一致.)
执行流程:- 判断线程池 的那个线程 是否空闲
- 空闲则使用
- 不空闲,则等待 池中的单个线程空闲后 使用
代码如下:
ExecutorService service = Executors.newSingleThreadExecutor();
service.execute(new Runnable() {
@Override
public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
service.execute(new Runnable() {
@Override public void run() {
System.out.println("线程的名称:"+Thread.currentThread().getName());
}
});
-
周期性任务定长线程池(可以周期执行)
执行流程:- 判断线程池是否存在空闲线程
- 存在则使用
- 不存在空闲线程,且线程池未满的情况下,则创建线程 并放入线程池, 然后使用
- 不存在空闲线程,且线程池已满的情况下,则等待线程池存在空闲线程
- 周期性任务执行时: 定时执行, 当某个时机触发时, 自动执行某任务 .
代码如下:
ScheduledExecutorService service = Executors.newScheduledThreadPool(2);
/*** 定时执行
* 参数1. runnable类型的任务
* 参数2. 时长数字
* 参数3. 时长数字的单位
*/
/*service.schedule(new Runnable() {
@Override
public void run() {
System.out.println("俩人相视一笑~ 嘿嘿嘿");
}
},5,TimeUnit.SECONDS);
*/
/*** 周期执行
* 参数1. runnable类型的任务
* 参数2. 时长数字(延迟执行的时长)
* 参数3. 周期时长(每次执行的间隔时间)
* 参数4. 时长数字的单位
*/
service.scheduleAtFixedRate(new Runnable() {
@Override
public void run() {
System.out.println("俩人相视一笑~ 嘿嘿嘿");
}
},5,2,TimeUnit.SECONDS);
}