分而治之—Fork-Join

Fork-Join

java下多線程的開發(fā)可以我們自己?jiǎn)⒂枚嗑€程,線程池撒强,還可以使用forkjoin,forkjoin可以讓我們不去了解諸如Thread,Runnable等相關(guān)的知識(shí),只要遵循forkjoin的開發(fā)模式漱办,就可以寫出很好的多線程并發(fā)程序。
Fork-Join使用分治發(fā)的思想婉烟,將一個(gè)難以直接解決的大問題娩井,分割成一些規(guī)模較小的相同問題,以便各個(gè)擊破似袁,分而治之洞辣。常見的歸并排序、二分查找昙衅、快速排序都是扬霜。

Fork-Join原理

圖片 1.png

Fork-Join框架就是在必要的情況下,將一個(gè)任務(wù)進(jìn)行拆分fork成若干的小任務(wù)再將一個(gè)小任務(wù)運(yùn)算結(jié)果進(jìn)行join匯總而涉。

工作密取
圖片 2.png
    private static class ConsumerAndProducer implements Runnable{  
        private Random random=new Random();  
        private final LinkedBlockingDeque<Work> deque;  
        private final LinkedBlockingDeque<Work> otherWork;  
        public ConsumerAndProducer(LinkedBlockingDeque<Work> deque,LinkedBlockingDeque<Work> otherWork){  
            this.deque = deque;  
            this.otherWork = otherWork;  
        }  
        @Override  
        public void run() {  
            //dosomethting
             if(deque.isEmpty()){  
                        if(!otherWork.isEmpty()){  
                            System.out.println("otherWork is run:");
                            otherWork.takeLast().run();;  
                        }  
                          
            }
        }
        
    }

即當(dāng)前線程的Task已經(jīng)全部執(zhí)行完畢著瓶,則自動(dòng)取到其他線程的Task池中的Task執(zhí)行。ForkJoinPool中維護(hù)著多個(gè)線程(一般為CPU核數(shù))在不斷地執(zhí)行Task婴谱,每個(gè)線程除了執(zhí)行自己職務(wù)內(nèi)的Task之外蟹但,還會(huì)根據(jù)自己工作線程的閑置情況去獲取其他繁忙的工作線程的Task,如此一來就能能夠減少線程阻塞或是閑置的時(shí)間谭羔,提高CPU利用率华糖。代碼上沒有什么好說的,利用雙端隊(duì)列將繁忙的線程的任務(wù)拿到自己線程去處理瘟裸。

Fork-Join實(shí)戰(zhàn)

圖片 3.png

我們要使用ForkJoin框架客叉,必須首先創(chuàng)建一個(gè)ForkJoin任務(wù)。它提供在任務(wù)中執(zhí)行fork和join的操作機(jī)制,通常我們不直接繼承ForkjoinTask類兼搏,只需要直接繼承其子類卵慰。

  1. RecursiveAction,用于沒有返回結(jié)果的任務(wù)
  2. RecursiveTask佛呻,用于有返回值的任務(wù)
    task要通過ForkJoinPool來執(zhí)行裳朋,使用submit 或 invoke 提交,兩者的區(qū)別是:invoke是同步執(zhí)行吓著,調(diào)用之后需要等待任務(wù)完成鲤嫡,才能執(zhí)行后面的代碼;submit是異步執(zhí)行绑莺。

在我們自己實(shí)現(xiàn)的compute方法里暖眼,首先需要判斷任務(wù)是否足夠小,如果足夠小就直接執(zhí)行任務(wù)纺裁。如果不足夠小诫肠,就必須分割成兩個(gè)子任務(wù),每個(gè)子任務(wù)在調(diào)用invokeAll方法時(shí)欺缘,又會(huì)進(jìn)入compute方法栋豫,看看當(dāng)前子任務(wù)是否需要繼續(xù)分割成孫任務(wù),如果不需要繼續(xù)分割浪南,則執(zhí)行當(dāng)前子任務(wù)并返回結(jié)果笼才。使用join方法會(huì)等待子任務(wù)執(zhí)行完并得到其結(jié)果。
例如我們要給一個(gè)文件夾中包含子文件夾下的文件某些文件

public class FindDirsFiles extends RecursiveAction {

    private File path;

    public FindDirsFiles(File path) {
        this.path = path;
    }

    @Override
    protected void compute() {
        List<FindDirsFiles> subTasks = new ArrayList<>();

        File[] files = path.listFiles();
        if (files!=null){
            for (File file : files) {
                if (file.isDirectory()) {
                    // 對(duì)每個(gè)子目錄都新建一個(gè)子任務(wù)络凿。
                    subTasks.add(new FindDirsFiles(file));
                } else {
                    // 遇到文件骡送,檢查。
                    if (file.getAbsolutePath().endsWith("txt")){
                        System.out.println("文件:" + file.getAbsolutePath());
                    }
                }
            }
            if (!subTasks.isEmpty()) {
                // 在當(dāng)前的 ForkJoinPool 上調(diào)度所有的子任務(wù)絮记。
                for (FindDirsFiles subTask : invokeAll(subTasks)) {
                    subTask.join();
                }
            }
        }
    }

    public static void main(String [] args){
        try {
            // 用一個(gè) ForkJoinPool 實(shí)例調(diào)度總?cè)蝿?wù)
            ForkJoinPool pool = new ForkJoinPool();
            FindDirsFiles task = new FindDirsFiles(new File("F:/"));

            /*異步提交*/
            pool.execute(task);

            /*主線程做自己的業(yè)務(wù)工作*/
            System.out.println("Task is Running......");
            Thread.sleep(1);
            int otherWork = 0;
            for(int i=0;i<100;i++){
                otherWork = otherWork + i;
            }
            System.out.println("Main Thread done sth......,otherWork="
                    +otherWork);
            task.join();//阻塞方法
            System.out.println("Task end");
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

我們首先初始化連接池摔踱,遍歷當(dāng)前文件下的文件夾存放到subTasks集合中,通過invokeAll方法放入ForkJoinPool中怨愤,遍歷每個(gè)FindDirsFiles的join獲得結(jié)果派敷。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市撰洗,隨后出現(xiàn)的幾起案子篮愉,更是在濱河造成了極大的恐慌,老刑警劉巖差导,帶你破解...
    沈念sama閱讀 216,470評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件试躏,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡设褐,警方通過查閱死者的電腦和手機(jī)颠蕴,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,393評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門泣刹,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人犀被,你說我怎么就攤上這事椅您。” “怎么了寡键?”我有些...
    開封第一講書人閱讀 162,577評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵掀泳,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我昌腰,道長(zhǎng)开伏,這世上最難降的妖魔是什么膀跌? 我笑而不...
    開封第一講書人閱讀 58,176評(píng)論 1 292
  • 正文 為了忘掉前任遭商,我火速辦了婚禮,結(jié)果婚禮上捅伤,老公的妹妹穿的比我還像新娘劫流。我一直安慰自己,他們只是感情好丛忆,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,189評(píng)論 6 388
  • 文/花漫 我一把揭開白布祠汇。 她就那樣靜靜地躺著,像睡著了一般熄诡。 火紅的嫁衣襯著肌膚如雪可很。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,155評(píng)論 1 299
  • 那天凰浮,我揣著相機(jī)與錄音我抠,去河邊找鬼。 笑死袜茧,一個(gè)胖子當(dāng)著我的面吹牛菜拓,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播笛厦,決...
    沈念sama閱讀 40,041評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼纳鼎,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了裳凸?” 一聲冷哼從身側(cè)響起贱鄙,我...
    開封第一講書人閱讀 38,903評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎姨谷,沒想到半個(gè)月后逗宁,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,319評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡菠秒,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,539評(píng)論 2 332
  • 正文 我和宋清朗相戀三年疙剑,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了氯迂。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,703評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡言缤,死狀恐怖嚼蚀,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情管挟,我是刑警寧澤轿曙,帶...
    沈念sama閱讀 35,417評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站僻孝,受9級(jí)特大地震影響导帝,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜穿铆,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,013評(píng)論 3 325
  • 文/蒙蒙 一您单、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧荞雏,春花似錦虐秦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,664評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至筑辨,卻和暖如春俺驶,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背棍辕。 一陣腳步聲響...
    開封第一講書人閱讀 32,818評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工暮现, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人痢毒。 一個(gè)月前我還...
    沈念sama閱讀 47,711評(píng)論 2 368
  • 正文 我出身青樓送矩,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親哪替。 傳聞我的和親對(duì)象是個(gè)殘疾皇子栋荸,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,601評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容

  • 2017年4月23日 星期日 晴 早上早起學(xué)習(xí),然后讀經(jīng)典凭舶。 上午9點(diǎn)出發(fā)到仁壽縣城參加第110期戶外學(xué)經(jīng)班晌块。很久...
    三寶媽肖雪強(qiáng)閱讀 215評(píng)論 0 0
  • 【配樂:原諒我就是這樣的女生——戴佩妮】 我想給自己起個(gè)名字,叫“框框姐姐”帅霜。 因?yàn)槲易鍪裁词露紩?huì)給自己畫個(gè)框匆背。自...
    夏日開的花閱讀 467評(píng)論 0 2
  • 霍華德·加德納多元智能理論給教師的啟示 讀了伍爾付凼克《教育心理學(xué)》第4章括享,讓我們?yōu)閹熣咴敿?xì)了解了學(xué)習(xí)者...
    李躍鵬閱讀 223評(píng)論 0 3