• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆 - 55  文章 - 15  trackbacks - 0
            <2012年5月>
            293012345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            常用鏈接

            留言簿

            隨筆分類

            隨筆檔案

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

                  ZThread中,有個抽象基類Runnable,提供了一個公共接口來執(zhí)行任務:

            class Runnable
            {
              public:
                virtual void run() = 0;
                vitural ~Runnable(){}
            };

                  只要從這個類繼承,并且重寫run()函數,就可以定義一個任務了!!!即具備了多線程的基礎。

                  但是只是定義了一個任務,它并不具有線程處理的能力。還要推它一把。

                  要使用一個線程,必須要先初始化一個線程,并用Runnable* 類型的對象來構造該線程。該線程在構造函數中會自動調用run()函數,此時,main()函數與run()函數齊飛,秋水共長天一色。

            請時刻記住,多線程的程序跟普通程序有不同之處,請看下面一段代碼

            class test : public Runnable
            {
            int id;
            public:
               test(int i = 0) : id(i){}
               void run()
                  {
                      cout <<" Do Something " << endl;
                  }
            };
            int main()
            {
             try{
               forint i = 0; i < 5; i++ )
                 Thread t(new test(i));//看這里看這里,小痘痘沒有了!!!
                }catch(Synchronization_Exception& e){
                 cerr << e.what() << endl;
                }
            }

            按理說,t是個局部變量,每次循環(huán),t都會被自動釋放掉。這里會不會這個線程對象就沒有了呢?從結果上來看,答案是,不會!從原理上來看是,當創(chuàng)建了一個Thread對象的時候,相關聯的線程會在線程處理系統(tǒng)中自動注冊,并保持其活動狀態(tài)。基于棧的對象被拋棄,但是線程本身還是活在線程處理系統(tǒng)中。

                   如果這么說的話,那么這種情況就有點像,t是一個引用,指向線程處理系統(tǒng)中的實際對象。引用被咔嚓掉,實例還在。從結果上看是這樣的。我這么解釋,是便于理解。

            一個線程想做兩個任務的話是不行滴,做了個很無聊的實驗,想看看線程處理系統(tǒng)中的實際對象是個嘛
            Thread t(new test(1));
            Thread t(new test(1));
            //----------華麗麗的分割線-----------錯誤,t重定義了//

            {
              Thread t(new test(1));
            }
            {
              Thread t(new test(1));
            }
            //----------華麗麗的分割線-----------正常,兩個線程, 兩個不同的任務//

            test* tp = new test;// same task
            Thread t1(tp);
            Thread t2(tp);
            //----------華麗麗的分割線-----------非常正常,兩個線程完成一個任務// 
            以上實驗說明,棧上的名字真的不重要!!!

            通過多線程,可以創(chuàng)建一個有響應的用戶界面。具體程序就不在碼了,這里要說的是有兩個問題:1. 怎么退出一個線程 2. 線程間通信。
            1. 怎么退出?正常退出,強制退出(包括中斷),帶來的問題是,非正常退出的話資源不不會釋放?
            2. 線程間通信,發(fā)送一個信號,讓子線程正常退出。

            考慮下面這段代碼:
            test* tp = new test;
            Thread t1(tp);
            delete tp;// 這相當于釜底抽薪,太絕了

            千萬別這么做,除非你跟你的公司有不共戴天之仇。這么做的問題在于,你不知道你的線程正在用tp做多么重要的事情,突然tp就這么沒了,線程也會傻在那里。最終導致代碼的不穩(wěn)定。不穩(wěn)定懂么,公司都需要穩(wěn)定的代碼。
            那么想要一個線程停下來,就讓它自己決定是否應該停,這時候它會清理現場,釋放堆棧之類的,總之,妥妥的。
            總結就是:給他一個flag,讓他自己決定是否退出,千萬別隨便決定別人的命運。

            使用執(zhí)行器簡化工作
                  其實沒看出來有什么簡化的,就是不用Thread的構造函數了,用一個executor.execute(Runnable*)函數來做,該函數的參數也是一個Runnable類型的指針。語義上比較好理解了吧。還沒有理解其真正的涵義。

                  執(zhí)行器每次都要創(chuàng)建線程,比較耗時耗資源。另外一種方法就是一次創(chuàng)建一堆線程,等你要用的時候就從線程池里面去取。該方法用PoolExecutor,示例代碼:
            #include "zthread/PoolExecutor.h"
            ..
            ..
            ..
            PoolExecutor executor(5);
            for(int i = 0; i < 5; i++)
            executor.execute(new test(i));
            ..
            ..
            concurrentExecutor 所有的任務用一個線程,當一個任務執(zhí)行完畢之后,后一個任務才開始執(zhí)行,示例代碼:
            #include"zthread/ConcurrentExecutor.h"
            //..
            ConcurrentExecutor executor;
            for(int i = 0; i < 5; i++ )
            executor.execute(new test(i));//任務按提交順序執(zhí)行,在下一個任務開始之前執(zhí)行完成
            字體越來越小了      腫么回事

            讓步
            yield()告訴CPU, 我做完了要做的事情,你可以讓別人跑了。
            sleep()告訴CPU,  我累了,睡會覺,你可以讓別人先跑。
            設置優(yōu)先級,告訴CPU,我們是有等級的,讓大佬先跑!!!代碼示例:

            #include "zhread/Thread.h"
            //..Something else 
            Thread high(new test(1));
            Thread medium(new test(2));
            Thread low(new test(3));
            high.setPriority(High);//總舵主
            medium.setPriority(Medium);//分舵主
            low.setPriority(Low);//成員

            哎,有了等級,大家就更有順序了“讓領導先走!!!”

            總結:創(chuàng)建一個任務很簡單,重寫run就對了。讓一個線程執(zhí)行一個任務很簡單,構造函數中放個任務指針就對了。想要輕松執(zhí)行線程很簡單,用個executor幫你就對了。領導先走很簡單,設個優(yōu)先級就對了。

            下一章:共享有限資源。

            posted on 2012-05-15 17:22 Dino-Tech 閱讀(505) 評論(1)  編輯 收藏 引用

            FeedBack:
            # re: Thinking in c++ -- 并發(fā)(1)定義任務,使用線程 2012-07-19 15:12 qq675976614
            同一個任務不可以被兩個線程完成的吧,線程會把它delete掉的 不管它是棧變量還是堆變量  回復  更多評論
              
            久久精品天天中文字幕人妻| 欧美成a人片免费看久久| 99久久做夜夜爱天天做精品| 久久久久亚洲av成人无码电影| 国产精品久久自在自线观看| 亚洲精品乱码久久久久久蜜桃图片 | 久久无码专区国产精品发布| 中文字幕亚洲综合久久菠萝蜜| 欧美无乱码久久久免费午夜一区二区三区中文字幕 | 久久人人爽人人爽AV片| 久久亚洲国产精品五月天婷| 欧美激情精品久久久久久久| 久久精品中文无码资源站| 久久婷婷激情综合色综合俺也去| 久久久久高潮毛片免费全部播放| 久久精品国产亚洲AV无码麻豆| 精品国产一区二区三区久久蜜臀| 久久国产精品视频| 亚洲精品无码专区久久久| 久久93精品国产91久久综合| 色偷偷久久一区二区三区| 青青青国产成人久久111网站| 久久精品成人免费观看97| 亚洲欧美日韩中文久久| 久久久久久免费一区二区三区 | 国产农村妇女毛片精品久久| 免费无码国产欧美久久18| 国产亚洲欧美成人久久片| 午夜精品久久久久久影视777 | 综合久久给合久久狠狠狠97色| 久久亚洲精品无码AV红樱桃| 久久精品成人影院| 999久久久免费精品国产| 亚洲精品NV久久久久久久久久 | 久久久久国产一级毛片高清版| 香蕉久久久久久狠狠色| 国产美女久久久| 精品综合久久久久久888蜜芽| 久久精品亚洲AV久久久无码| 国产精品久久久久久久午夜片| 久久久久AV综合网成人|