多线程之:模拟实现线程池的工作原理

[一]线程池存在的价值:

==>多线程技术主要解决处理器单元内多个线程执行的问题,它可以显著减少处理器单元的闲置时间,增加处理器单元的吞吐能力。   
==>假设一个服务器完成一项任务所需时间为: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 } 
View Code
原文地址:https://www.cnblogs.com/shangxiaofei/p/5487119.html