Java如何固定大小的線程池
線程池就是在程序啟動(dòng)的時(shí)候先建立幾個(gè)可以使用的線程放在那里,然后等著具體的任務(wù)放進(jìn)去,這個(gè)任務(wù)基本可以說(shuō)都是Runnable的實(shí)現(xiàn)類,因此它減小了系統(tǒng)每次新建和銷毀線程的開(kāi)銷,但同時(shí)增加了維護(hù)這些線程的開(kāi)銷,個(gè)中取舍看具體情況而定。
固定大小的線程池就是在啟動(dòng)的時(shí)候創(chuàng)建了固定個(gè)數(shù)的線程放在那里等待使用。
2.包裝一個(gè)線程池對(duì)象public class TaskPool{ private final ThreadPoolExecutor executor = (ThreadPoolExecutor)Executors.newFixedThreadPool(9); // 創(chuàng)建一個(gè)大小為9的固定線程池,可以按照CPU的核數(shù)初步判定,如果CPU密集性任務(wù)則創(chuàng)建N+1個(gè),如果是IO密集型任務(wù)則創(chuàng)建2N+1個(gè),其中N即CPU的核數(shù) protected void shutdown(){// do something// 這個(gè)方法等待線程池中所有已提交任務(wù)執(zhí)行結(jié)束,不接收新任務(wù),然后結(jié)束executor.shutdown(); // 這個(gè)強(qiáng)制結(jié)束所有任務(wù),然后正在等在的任務(wù)列表// executor.shutdownNow(); } protected void execute(Runnable command){// do something// 提交任務(wù)executor.execute(command); } public void status(){StringBuffer sb = new StringBuffer();// 當(dāng)前正在執(zhí)行任務(wù)的線程數(shù)sb.append(executor.getActiveCount() + 'n'); // 當(dāng)前正在等待執(zhí)行的線程數(shù)sb.append(executor.getQueue().size() + 'n'); // 返回已經(jīng)完成的線程數(shù)sb.append(executor.getCompletedTaskCount() + 'n'); System.out.println(sb.toString());// 注:以上方法都是返回一個(gè)大概值,因?yàn)榫€程在執(zhí)行中,這些狀態(tài)隨時(shí)都會(huì)改變 }} 3.使用線程池
public class Launcher{ private TaskPool taskPool = new TaskPool(); public static void main(String[] args){// 新建100個(gè)任務(wù),Runnable的實(shí)現(xiàn)類TaskTask[] tasks = new Task[100];for (int i = 0; i < tasks.length; i++){ tasks[i] = new Task('Task ' + (i+1)); // 提交到線程池運(yùn)行 taskPool.execute(task[i]); if ( i % 50 == 0){taskPool.status();} } private static class Task implements Runnable{private String name;public Task(String name){ this.name = name;}public void run(){ // do something System.out.println('我的名字是:' + this.name);} }}Java線程池小拓展線程池的介紹
1 常用的 池化技術(shù)
C3P0
DBCP
2 線程池的衍生
頻繁的創(chuàng)建線程對(duì)象和多線程之間進(jìn)行上下文切換,是非常耗時(shí)間和資源的所以JDK1.5中提出了線程池技術(shù)
3 使用線程池
Exector
4 線程池的創(chuàng)建
創(chuàng)建一個(gè)固定大小的線程池 ( 最常用的方法 )
ExecutorService pool = Executors.newFixedThreadPool(2);Runnable task = new Runnable() {@Overridepublic void run() {while (true) {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName());}}};pool.execute(task);pool.execute(task);pool.execute(task);//線程池的帶下只有兩個(gè) 現(xiàn)在這個(gè)任務(wù)在其等待隊(duì)列中排隊(duì)等候
創(chuàng)建可變大小的線程池
ExecutorService pool = Executors.newCachedThreadPool();Runnable task = new Runnable() {@Overridepublic void run() {while (true) {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName());}}};pool.execute(task);pool.execute(task);pool.execute(task);
創(chuàng)建獨(dú)立任務(wù)的線程池
ExecutorService pool = Executors.newSingleThreadExecutor();Runnable task = new Runnable() {@Overridepublic void run() {while (true) {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName());}}};pool.execute(task);pool.execute(task);pool.execute(task);
創(chuàng)建可調(diào)度的線程池
ScheduledExecutorService threadPool = Executors.newScheduledThreadPool(2);Runnable task = new Runnable() {@Overridepublic void run() {while (true) {try {Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}System.out.println(Thread.currentThread().getName());}}};threadPool.schedule(task, 2000, TimeUnit.MILLISECONDS);
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
1. Warning: require(): open_basedir restriction in effect,目錄配置open_basedir報(bào)錯(cuò)問(wèn)題分析2. ASP中常用的22個(gè)FSO文件操作函數(shù)整理3. php網(wǎng)絡(luò)安全中命令執(zhí)行漏洞的產(chǎn)生及本質(zhì)探究4. ASP的Global.asa文件技巧用法5. php測(cè)試程序運(yùn)行速度和頁(yè)面執(zhí)行速度的代碼6. html清除浮動(dòng)的6種方法示例7. SharePoint Server 2019新特性介紹8. ASP中if語(yǔ)句、select 、while循環(huán)的使用方法9. React+umi+typeScript創(chuàng)建項(xiàng)目的過(guò)程10. Vue+elementUI下拉框自定義顏色選擇器方式
