多线程之:模拟实现线程池的工作原理
[一]线程池存在的价值:
==>多线程技术主要解决处理器单元内多个线程执行的问题,它可以显著减少处理器单元的闲置时间,增加处理器单元的吞吐能力。
==>假设一个服务器完成一项任务所需时间为:T1 创建线程时间,T2 在线程中执行任务的时间,T3 销毁线程时间。
==>如果:T1 + T3 远大于 T2,则可以采用线程池,以提高服务器性能。
[二]合理利用线程池能够带来三个好处。
* 第一:降低资源消耗。通过重复利用已创建的线程降低线程创建和销毁造成的消耗。
* 第二:提高响应速度。当任务到达时,任务可以不需要等到线程创建就能立即执行。
* 第三:提高线程的可管理
[三]一个线程池的组成部分
(1)线程池管理器
=>其中线程池管理器的作用是创建、销毁并管理线程池,将工作线程放入线程池中;
=>线程池管理器至少有下列功能:创建线程池,销毁线程池,添加新任务。
(2)工作线程
=>工作线程是一个可以循环执行任务的线程,在没有任务是进行等待;
=>工作线程是一个可以循环执行任务的线程,在没有任务时将等待。
(3)任务列队
=>任务列队的作用是提供一种缓冲机制,将没有处理的任务放在任务列队中;
(4)任务接口等部分。
=>任务接口是每个任务必须实现的接口,主要用来规定任务的入口、任务执行完后的收尾工作、任务的执行状态等,工作线程通过该接口调度任务的执行。
=>任务接口是为所有任务提供统一的接口,以便工作线程处理。任务接口主要规定了任务的入口,任务执行完后的收尾工作,任务的执行状态等。
[四]模拟实现一个线程池的原理
1 package mine.util.thread; 2 3 import java.util.LinkedList; 4 import java.util.List; 5 6 /** 7 * 线程池类,线程管理器:创建线程,执行任务,销毁线程,获取线程基本信息 8 */ 9 public final class ThreadPool { 10 // 线程池中默认线程的个数为5 11 private static int worker_num = 5; 12 // 工作线程 13 private WorkThread[] workThrads; 14 // 未处理的任务 15 private static volatile int finished_task = 0; 16 // 任务队列,作为一个缓冲,List线程不安全 17 private List<Runnable> taskQueue = new LinkedList<Runnable>(); 18 private static ThreadPool threadPool; 19 20 // 创建具有默认线程个数的线程池 21 private ThreadPool() { 22 this(5); 23 } 24 25 // 创建线程池,worker_num为线程池中工作线程的个数 26 private ThreadPool(int worker_num) { 27 ThreadPool.worker_num = worker_num; 28 workThrads = new WorkThread[worker_num]; 29 for (int i = 0; i < worker_num; i++) { 30 workThrads[i] = new WorkThread(); 31 workThrads[i].start();// 开启线程池中的线程 32 } 33 } 34 35 // 单态模式,获得一个默认线程个数的线程池 36 public static ThreadPool getThreadPool() { 37 return getThreadPool(ThreadPool.worker_num); 38 } 39 40 // 单态模式,获得一个指定线程个数的线程池,worker_num(>0)为线程池中工作线程的个数 41 // worker_num<=0创建默认的工作线程个数 42 public static ThreadPool getThreadPool(int worker_num1) { 43 if (worker_num1 <= 0) 44 worker_num1 = ThreadPool.worker_num; 45 if (threadPool == null) 46 threadPool = new ThreadPool(worker_num1); 47 return threadPool; 48 } 49 50 // 执行任务,其实只是把任务加入任务队列,什么时候执行有线程池管理器觉定 51 public void execute(Runnable task) { 52 synchronized (taskQueue) { 53 taskQueue.add(task); 54 taskQueue.notify(); 55 } 56 } 57 58 // 批量执行任务,其实只是把任务加入任务队列,什么时候执行有线程池管理器觉定 59 public void execute(Runnable[] task) { 60 synchronized (taskQueue) { 61 for (Runnable t : task) 62 taskQueue.add(t); 63 taskQueue.notify(); 64 } 65 } 66 67 // 批量执行任务,其实只是把任务加入任务队列,什么时候执行有线程池管理器觉定 68 public void execute(List<Runnable> task) { 69 synchronized (taskQueue) { 70 for (Runnable t : task) 71 taskQueue.add(t); 72 taskQueue.notify(); 73 } 74 } 75 76 // 销毁线程池,该方法保证在所有任务都完成的情况下才销毁所有线程,否则等待任务完成才销毁 77 public void destroy() { 78 while (!taskQueue.isEmpty()) {// 如果还有任务没执行完成,就先睡会吧 79 try { 80 Thread.sleep(10); 81 } catch (InterruptedException e) { 82 e.printStackTrace(); 83 } 84 } 85 // 工作线程停止工作,且置为null 86 for (int i = 0; i < worker_num; i++) { 87 workThrads[i].stopWorker(); 88 workThrads[i] = null; 89 } 90 threadPool=null; 91 taskQueue.clear();// 清空任务队列 92 } 93 94 // 返回工作线程的个数 95 public int getWorkThreadNumber() { 96 return worker_num; 97 } 98 99 // 返回已完成任务的个数,这里的已完成是只出了任务队列的任务个数,可能该任务并没有实际执行完成 100 public int getFinishedTasknumber() { 101 return finished_task; 102 } 103 104 // 返回任务队列的长度,即还没处理的任务个数 105 public int getWaitTasknumber() { 106 return taskQueue.size(); 107 } 108 109 // 覆盖toString方法,返回线程池信息:工作线程个数和已完成任务个数 110 @Override 111 public String toString() { 112 return "WorkThread number:" + worker_num + " finished task number:" 113 + finished_task + " wait task number:" + getWaitTasknumber(); 114 } 115 116 /** 117 * 内部类,工作线程 118 */ 119 private class WorkThread extends Thread { 120 // 该工作线程是否有效,用于结束该工作线程 121 private boolean isRunning = true; 122 123 /* 124 * 关键所在啊,如果任务队列不空,则取出任务执行,若任务队列空,则等待 125 */ 126 @Override 127 public void run() { 128 Runnable r = null; 129 while (isRunning) {// 注意,若线程无效则自然结束run方法,该线程就没用了 130 synchronized (taskQueue) { 131 while (isRunning && taskQueue.isEmpty()) {// 队列为空 132 try { 133 taskQueue.wait(20); 134 } catch (InterruptedException e) { 135 e.printStackTrace(); 136 } 137 } 138 if (!taskQueue.isEmpty()) 139 r = taskQueue.remove(0);// 取出任务 140 } 141 if (r != null) { 142 r.run();// 执行任务 143 } 144 finished_task++; 145 r = null; 146 } 147 } 148 149 // 停止工作,让该线程自然执行完run方法,自然结束 150 public void stopWorker() { 151 isRunning = false; 152 } 153 } 154 }