這篇文章主要為大家展示了“Java如何實現手寫線程池”,內容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領大家一起研究并學習一下“Java如何實現手寫線程池”這篇文章吧。
目前創新互聯已為超過千家的企業提供了網站建設、域名、雅安服務器托管、網站托管、企業網站設計、盤州網站維護等服務,公司將堅持客戶導向、應用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協力一起成長,共同發展。
具體內容如下
1.線程池是一種多線程處理形式,處理過程中將任務添加到隊列,然后在創建線程后自動啟動這些任務。線程池線程都是后臺線程。
2.線程池簡易架構
3.簡易線程池代碼(自行優化)
import java.util.List; /** * 線程接口 * * @Author yjian * @Date 14:49 2017/10/14 **/ public interface IThreadPool { //加入任務 void execute(Runnable task); //加入任務 void execute(Runnable[] tasks); //加入任務 void execute(List<Runnable> tasks); //銷毀線程 void destroy(); }
import java.util.LinkedList; import java.util.List; import java.util.concurrent.atomic.AtomicLong; /** * 線程實現類(簡易實現,自行優化.提供思路) * * @Author yjian * @Date 14:49 2017/10/14 **/ @SuppressWarnings("ALL") public class ThreadPoolImpl implements IThreadPool { //默認開啟線程個數 static int WORKER_NUMBER = 5; //完成任務線程數 可見性 static volatile int sumCount = 0; //任務隊列 list非線程安全,可以優化為BlockingQueue static List<Runnable> taskQueue = new LinkedList<Runnable>(); //線程工作組 WorkerThread[] workThreads; //原子性 static AtomicLong threadNum = new AtomicLong(); static ThreadPoolImpl threadPool; //構造方法 public ThreadPoolImpl() { this(WORKER_NUMBER); } public ThreadPoolImpl(int workerNum) { this.WORKER_NUMBER = workerNum; //開辟工作線程空間 workThreads = new WorkerThread[WORKER_NUMBER]; //開始創建工作線程 for (int i = 0; i < WORKER_NUMBER; i++) { workThreads[i] = new WorkerThread(); Thread thread = new Thread(workThreads[i], "ThreadPool-worker" + threadNum.incrementAndGet()); System.out.println("初始化線程數" + (i + 1) + "---------當前線程名稱:" + thread.getName()); thread.start(); } } @Override public String toString() { return "工作線程數量為" + WORKER_NUMBER + "已完成的任務數" + sumCount + "等待任務數量" + taskQueue.size(); } //獲取線程池 public static IThreadPool getThreadPool() { return getThreadPool(WORKER_NUMBER); } public static IThreadPool getThreadPool(int workerNum) { //容錯性,如果小于等于0就默認線程數 if (workerNum <= 0) { workerNum = WORKER_NUMBER; } if (threadPool == null) { threadPool = new ThreadPoolImpl(workerNum); } return threadPool; } @Override public void execute(Runnable task) { synchronized (taskQueue) { taskQueue.add(task); taskQueue.notifyAll(); } } @Override public void execute(Runnable[] tasks) { synchronized (taskQueue) { for (Runnable task : tasks) { taskQueue.add(task); } taskQueue.notifyAll(); } } @Override public void execute(List<Runnable> tasks) { synchronized (taskQueue) { for (Runnable task : tasks) { taskQueue.add(task); } taskQueue.notifyAll(); } } @Override public void destroy() { //循環是否還存在任務,如果存在等待20毫秒處理時間 while (!taskQueue.isEmpty()) { try { Thread.sleep(20); } catch (InterruptedException e) { e.printStackTrace(); } } //如果任務隊列已處理完成,銷毀線程,清空任務 for (int i = 0; i < WORKER_NUMBER; i++) { workThreads[i].setWorkerFlag(); workThreads[i] = null; } threadPool = null; taskQueue.clear(); } //創建工作線程池 class WorkerThread extends Thread { //用來標識當前線程屬于活動可用狀態 private boolean isRunning = true; @Override public void run() { Runnable runnable = null; //死循環 while (isRunning) { //非線程安全,所以采用同步鎖 synchronized (taskQueue) { while (isRunning && taskQueue.isEmpty()) { try { //如果任務隊列為空,等待20毫秒 監聽任務到達 taskQueue.wait(20); } catch (Exception e) { e.printStackTrace(); } } //任務隊列不為空 if (!taskQueue.isEmpty()) { runnable = taskQueue.remove(0);//獲取第一個任務 } } if (runnable != null) { runnable.run(); } sumCount++; runnable = null; } } //銷毀線程 public void setWorkerFlag() { isRunning = false; } } }
import java.util.ArrayList; import java.util.List; /** * 測試類 * * @Author yjian * @Date 15:37 2017/10/14 **/ public class ThreadPoolTest { public static void main(String[] args) { //獲取線程池 IThreadPool t = ThreadPoolImpl.getThreadPool(20); List<Runnable> taskList = new ArrayList<Runnable>(); for (int i = 0; i < 100; i++) { taskList.add(new Task()); } //執行任務 t.execute(taskList); System.out.println(t); //銷毀線程 t.destroy(); System.out.println(t); } static class Task implements Runnable { private static volatile int i = 1; @Override public void run() { System.out.println("當前處理的線程:" + Thread.currentThread().getName() + " 執行任務" + (i++) + " 完成"); } } }
對spring源碼研究的,仔細查看代碼用了哪幾種spring常用的模式。寫程序的規范應該和spring一樣。
以上是“Java如何實現手寫線程池”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注創新互聯行業資訊頻道!
當前標題:Java如何實現手寫線程池
文章轉載:http://m.kartarina.com/article22/gecjcc.html
成都網站建設公司_創新互聯,為您提供網站導航、定制網站、面包屑導航、電子商務、微信公眾號、網站內鏈
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯