线程池是和数据库连接池类似的一种池,而仅仅是把池里的对象换成了线程。
核心思想:最主要就是复用的思想,把运行阶段尽量拉长,对每个任务的到来,不是重复创建、销毁,而是重复利用之前建立的线程来执行任务
线程池的作用:
在程序启动的时候就创建若干线程来响应处理,他们被称为线程池,里面的线程称之为工作线程。
1.降低资源损耗。通过重复利用已经创建好的线程来减少在创建新线程和销毁的时候带来的资源损耗
2.提高响应速度,当新任务到来时,可以立即执行,而不用等待新线程的建立
3.提高线程的可管理性
一个线程池的属性起码包括初始化线程数量、线程数组、任务队列。初始化线程数量指线程池初始化创建的线程数,线程数组保存了线程池中的所有线程,任务队列指添加到线程池中等待处理所有的任务。
执行如下图所示,池中有两个线程,池里线程的工作就是不断循环检测任务队列中是否有需要执行的任务,如果有,则处理并移出队列;因为普通线程的话就是执行完run方法之后就会被JVM的垃圾回收器所回收;循环检测就避免了这种情况,使线程一直存活。
简单实现:
public final class ThreadPool {
private final int workerThread_num;
private WorkerThread[] workerThreads;
private List<Runnable> taskQueue = new LinkedList<Runnable>();
private static ThreadPool threadPool;
public ThreadPool(int worker_num) {
this.workerThread_num = worker_num;
workerThreads = new WorkerThread[workerThread_num];
for (int i = 0; i < workerThread_num; i++) { // 启动工作线程
workerThreads[i] = new WorkerThread();
workerThreads[i].start();
}
}
public void execute(Runnable task) {
synchronized(taskQueue){
taskQueue.add(task);
}
}
private class WorkerThread extends Thread {
public void run() {
Runnable r = null;
while (true) { // 循环监听
synchronized (taskQueue) {
if (!taskQueue.isEmpty()) {
r = taskQueue.remove(0);
r.run(); // 执行任务
}
}
}
}
}
}
使用线程池时只需要实例化一个对象,构造函数就会创建相应数量的线程并启动线程,启动的线程开始无限循环检测任务队列,执行方法exe-cute()仅仅是把任务添加任务队列中。