java基礎:簡單實現(xiàn)線程池

前段時間自己研究了下線程池的實現(xiàn)原理,通過一些源碼對比扮饶,發(fā)現(xiàn)其實核心的東西不難夺巩,于是抽絲剝繭贞让,決定自己實現(xiàn)一個簡單線程池,當自已實現(xiàn)了出一個線程池后柳譬。發(fā)現(xiàn)原來那么高大上的東西也可以這么簡單喳张。

先上原理圖:為了更好的在手機上顯示,我重新把圖畫了一遍

線程池簡單實現(xiàn)

上代碼之前美澳,要先補充一下線程池構造的核心幾個點

  1. 線程池里的核心線程數(shù)與最大線程數(shù)
  2. 線程池里真正工作的線程worker
  3. 線程池里用來存取任務的隊列BlockingQueue
  4. 線程中的任務task

本例實現(xiàn)簡化了一些销部,只實現(xiàn)了BlockingQueue存放任務,然后每個worker取任務并執(zhí)行制跟,下面看代碼
首先定義一個線程池ThreadExcutor

class ThreadExcutor{

    //創(chuàng)建
    private volatile boolean RUNNING = true;

    //所有任務都放隊列中舅桩,讓工作線程來消費
    private static BlockingQueue<Runnable> queue = null;

    private final HashSet<Worker> workers = new HashSet<Worker>();

    private final List<Thread> threadList = new ArrayList<Thread>();

    //工作線程數(shù)
    int poolSize = 0;
    //核心線程數(shù)(創(chuàng)建了多少個工作線程)
    int coreSize = 0;

    boolean shutdown = false;

    public ThreadExcutor(int poolSize){
        this.poolSize = poolSize;
        queue = new LinkedBlockingQueue<Runnable>(poolSize);
    }

    public void exec(Runnable runnable) {
        if (runnable == null) throw new NullPointerException();
        if(coreSize < poolSize){
            addThread(runnable);
        }else{
            //System.out.println("offer" +  runnable.toString() + "   " + queue.size());
            try {
                queue.put(runnable);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public void addThread(Runnable runnable){
        coreSize ++;
        Worker worker = new Worker(runnable);
        workers.add(worker);
        Thread t = new Thread(worker);
        threadList.add(t);
        try {
            t.start();
        }catch (Exception e){
            e.printStackTrace();
        }

    }

    public void shutdown() {
        RUNNING = false;
        if(!workers.isEmpty()){
            for (Worker worker : workers){
                worker.interruptIfIdle();
            }
        }
        shutdown = true;
        Thread.currentThread().interrupt();
    }
   //這里留個位置放內部類Worker
 }

然后定義一個內部類Worker,這個內部類Worker是用來執(zhí)行每個任務的雨膨,在創(chuàng)建線程池后擂涛,往線程里添加任務,每個任務都是由Worker一個一個來啟動的聊记。

    /**
     * 工作線程
     */
    class  Worker implements Runnable{

        public Worker(Runnable runnable){
            queue.offer(runnable);
        }

        @Override
        public void run() {
            while (true && RUNNING){
                if(shutdown == true){
                    Thread.interrupted();
                }
                Runnable task = null;
                try {
                    task = getTask();
                    task.run();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }

        public Runnable getTask() throws InterruptedException {
            return queue.take();
        }

        public void interruptIfIdle() {
            for (Thread thread :threadList) {
                System.out.println(thread.getName() + " interrupt");
                thread.interrupt();
            }
        }
    }

首先注意的一點撒妈,這個Worker是個內部類,是在線程池內聲明的排监。

exec方法
只要將任務放到線程池定義的隊列中就行了
Worker怎么工作
任勞任怨的Worker

這個工作線程實例化的時候就先加入一個任務到隊列中狰右,也就是說在實例化這個工作線程時,這個工作線程也是一個任務被加入到線程池中舆床。然后就是run方法棋蚌,這個run方法是線程調start方法生成的線程嫁佳,而Worker調的run方法并沒有生成新的線程。就是一個循環(huán)谷暮,一直在不停的從隊列中取任務蒿往,然后執(zhí)行】辣福可以看到熄浓,取隊列的方法是take()情臭,這個方法意思如果隊列為空了省撑,取不到數(shù)據時就阻塞隊列。

然后看shutdown()
shutdown動作迅速

你每天辛勤的勞動著俯在,突然接收到上面的命令竟秫,說活暫時不要接了,先停下來跷乐,當你還沒搞清楚狀況時肥败,接著你的領導又把你開除了,說公司要倒了愕提,你先下崗吧馒稍,一會我也得下崗了。這就是shutdown做的事浅侨,shutdown必須是主線程才能停止工作線程纽谒。
shutdown方法并不是用線程那種強制停止的搞法,而是先用一個標識符告訴工作線程如输,不要再接任務了鼓黔。然后通知工作線程,你可以interrupt()了不见,當所有的線程停止后記得要把主線程也停掉澳化,這樣,一個簡單任務的線程池就完成了稳吮。
讓我們來測試一下:

/**
 * Created by wxwall on 2017/6/7.
 */
public class TheadBlockedQ {
    public static void main(String[] args) throws InterruptedException {
        ThreadExcutor excutor = new ThreadExcutor(3);
        for (int i = 0; i < 10; i++) {
            excutor.exec(new Runnable() {
                @Override
                public void run() {
                    System.out.println("線程 " + Thread.currentThread().getName() + " 在幫我干活");
                }
            });
        }
       excutor.shutdown();
    }
}

輸出結果為:

線程 Thread-0 在幫我干活
線程 Thread-2 在幫我干活
線程 Thread-1 在幫我干活
線程 Thread-0 在幫我干活
線程 Thread-2 在幫我干活
線程 Thread-2 在幫我干活
線程 Thread-1 在幫我干活
線程 Thread-0 在幫我干活
Thread-0 interrupt
Thread-1 interrupt
Thread-2 interrupt
Thread-0 interrupt
Thread-1 interrupt
Thread-2 interrupt
Thread-0 interrupt
Thread-1 interrupt
Thread-2 interrupt

這當然是最簡單實現(xiàn)缎谷,JDK的實現(xiàn)比這強大的多,而且還具備當工作線程處理不過來時灶似,可以產生新的線程來處理任務列林,這個數(shù)量不能超過原先定義的最大線程數(shù),而在本例中都沒實現(xiàn)這些功能喻奥。
我相信當想了解一個模塊的功能時席纽,如果一開始就了解其中最核心的點,然后向外慢慢擴展撞蚕,那么學習這個模塊時一定能省下不少時間润梯,而且理解將很深刻。希望這個簡單線程池實現(xiàn)能讓你有所領悟,以更加簡單的方式了解線程池纺铭,了解了線程池寇钉,對于其他池化技術,原理都是相通的舶赔。

最后我想說:我相信寫好一篇文章能讓大家理解不困難都是用了心的扫倡,您也點個贊支持支持。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末竟纳,一起剝皮案震驚了整個濱河市撵溃,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌锥累,老刑警劉巖缘挑,帶你破解...
    沈念sama閱讀 219,270評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異桶略,居然都是意外死亡语淘,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,489評論 3 395
  • 文/潘曉璐 我一進店門际歼,熙熙樓的掌柜王于貴愁眉苦臉地迎上來惶翻,“玉大人,你說我怎么就攤上這事鹅心÷来郑” “怎么了?”我有些...
    開封第一講書人閱讀 165,630評論 0 356
  • 文/不壞的土叔 我叫張陵巴帮,是天一觀的道長溯泣。 經常有香客問我,道長榕茧,這世上最難降的妖魔是什么垃沦? 我笑而不...
    開封第一講書人閱讀 58,906評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮用押,結果婚禮上肢簿,老公的妹妹穿的比我還像新娘。我一直安慰自己蜻拨,他們只是感情好池充,可當我...
    茶點故事閱讀 67,928評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著缎讼,像睡著了一般收夸。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上血崭,一...
    開封第一講書人閱讀 51,718評論 1 305
  • 那天卧惜,我揣著相機與錄音厘灼,去河邊找鬼。 笑死咽瓷,一個胖子當著我的面吹牛设凹,可吹牛的內容都是我干的。 我是一名探鬼主播茅姜,決...
    沈念sama閱讀 40,442評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼闪朱,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了钻洒?” 一聲冷哼從身側響起奋姿,我...
    開封第一講書人閱讀 39,345評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎航唆,沒想到半個月后胀蛮,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經...
    沈念sama閱讀 45,802評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡糯钙,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,984評論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了退腥。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片任岸。...
    茶點故事閱讀 40,117評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖狡刘,靈堂內的尸體忽然破棺而出享潜,到底是詐尸還是另有隱情,我是刑警寧澤嗅蔬,帶...
    沈念sama閱讀 35,810評論 5 346
  • 正文 年R本政府宣布剑按,位于F島的核電站,受9級特大地震影響澜术,放射性物質發(fā)生泄漏艺蝴。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,462評論 3 331
  • 文/蒙蒙 一鸟废、第九天 我趴在偏房一處隱蔽的房頂上張望猜敢。 院中可真熱鬧,春花似錦盒延、人聲如沸缩擂。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,011評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽胯盯。三九已至,卻和暖如春计露,著一層夾襖步出監(jiān)牢的瞬間博脑,已是汗流浹背楞捂。 一陣腳步聲響...
    開封第一講書人閱讀 33,139評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留趋厉,地道東北人寨闹。 一個月前我還...
    沈念sama閱讀 48,377評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像君账,于是被迫代替她去往敵國和親繁堡。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,060評論 2 355

推薦閱讀更多精彩內容