• <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>
            隨筆 - 96  文章 - 255  trackbacks - 0
            <2010年6月>
            303112345
            6789101112
            13141516171819
            20212223242526
            27282930123
            45678910

            E-mail:zbln426@163.com QQ:85132383 長期尋找對戰(zhàn)略游戲感興趣的合作伙伴。

            常用鏈接

            留言簿(21)

            隨筆分類

            隨筆檔案

            SDL相關(guān)網(wǎng)站

            我的個人網(wǎng)頁

            我的小游戲

            資源下載

            搜索

            •  

            積分與排名

            • 積分 - 493207
            • 排名 - 39

            最新評論

            閱讀排行榜

            評論排行榜

            還是先看一段非多線程的程序。我們用TestClass1表示在線程中創(chuàng)建的對象類,用TestClass2表示與創(chuàng)建線程的操作在同一定義域(也就是同一對{}之中的)的局部變量的對象類。
            之所以提出TestClass2這種類,是因為在實際編程中我們會遇到這種情況:我們不可預(yù)知這個類何時創(chuàng)建以及創(chuàng)建多少;這個類的對象是一個新線程的參數(shù)。比如,在sokcet中的TCP server端就會有這種情況:如果每一個新連接的client都用創(chuàng)建一個新的線程去處理,我們不可預(yù)知在什么時候會有多少客戶端連過來。
            我們先觀察沒有多線程的時候?qū)ο蟮纳芷冢?br>
            #include <iostream>
            #include 
            "windows.h"

            class TestClass1{
            private:
                
            int x;
            public:
                
            explicit TestClass1(int to_x):x(to_x)
                {}
                
            ~TestClass1()
                {
                    std::cerr 
            << "destruction: 1." << std::endl;
                }
                
            void show() const
                {
                    std::cerr 
            << x << std::endl;
                }
            };

            class TestClass2{
            private:
                
            int* pX;
            public:
                
            explicit TestClass2(int to_x)
                {
                    pX 
            = new int;
                    
            *pX = to_x;
                }
                
            ~TestClass2()
                {
                    delete pX;
                    std::cerr 
            << "destruction: 2."  << std::endl;
                }
                
            const int& value() const
                {
                    
            return *pX;
                }
            };

            DWORD WINAPI thread_func(LPVOID pN)
            {
                Sleep(
            200);
                TestClass1 test((
            *((TestClass2*)pN)).value());
                test.show();
                
            return 0;
            }

            int main(int argc, char* argv[])
            {
                
            for (int i = 0; i < 3++i) {
                    TestClass2 n(
            5);
                    
                    thread_func((LPVOID)
            &n);
                    std::cerr 
            << "loop: " << i+1 << std::endl;
                }

                Sleep(
            2000);

                std::cout 
            << "main() ok." << std::endl;

                
            return 0;
            }
            這是標準的C++模式,對象的生命周期是可以預(yù)見的:
            5
            destruction: 
            1.
            loop: 
            1
            destruction: 
            2.
            5
            destruction: 
            1.
            loop: 
            2
            destruction: 
            2.
            5
            destruction: 
            1.
            loop: 
            3
            destruction: 
            2.
            main() ok.
            請按任意鍵繼續(xù). . .
            如果我們改成線程調(diào)用:
            #include <iostream>
            #include 
            "windows.h"

            class TestClass1{
            private:
                
            int x;
            public:
                
            explicit TestClass1(int to_x):x(to_x)
                {}
                
            ~TestClass1()
                {
                    std::cerr 
            << "destruction: 1." << std::endl;
                }
                
            void show() const
                {
                    std::cerr 
            << x << std::endl;
                }
            };

            class TestClass2{
            private:
                
            int* pX;
            public:
                
            explicit TestClass2(int to_x)
                {
                    pX 
            = new int;
                    
            *pX = to_x;
                }
                
            ~TestClass2()
                {
                    delete pX;
                    std::cerr 
            << "destruction: 2."  << std::endl;
                }
                
            const int& value() const
                {
                    
            return *pX;
                }
            };

            DWORD WINAPI thread_func(LPVOID pN)
            {
                Sleep(
            200);
                TestClass1 test((
            *((TestClass2*)pN)).value());
                test.show();
                
            return 0;
            }

            int main(int argc, char* argv[])
            {
                
            for (int i = 0; i < 3++i) {
                    TestClass2 n(
            5);
                    
                    HANDLE hThrd;
                    DWORD thrdId;
                    hThrd 
            = CreateThread(    NULL,
                                            
            0,
                                            thread_func,
                                            (LPVOID)
            &n,
                                            
            0,
                                            
            &thrdId);
                    
                    std::cerr 
            << "loop: " << i+1 << std::endl;
                }

                Sleep(
            2000);

                std::cout 
            << "main() ok." << std::endl;

                
            return 0;
            }
            可以看到函數(shù)返回了錯誤的值(至于為什么每次都是36我還不清楚,但是至少不是正確的數(shù)字5),這是因為在線程調(diào)用TestClass2的對象之前已經(jīng)被析構(gòu)的緣故。
            loop: 1
            destruction: 
            2.
            loop: 
            2
            destruction: 
            2.
            loop: 
            3
            destruction: 
            2.
            36
            destruction: 
            1.
            36
            destruction: 
            1.
            36
            destruction: 
            1.
            main() ok.
            請按任意鍵繼續(xù). . .
            所以,如果我們設(shè)想構(gòu)造一個類,這個類的對象可以調(diào)用包含this的線程,那么這個對象一定不能是局部變量,或者說,我們必須在循環(huán)的{}對之前先把這些對象構(gòu)造出來。這與我們的需求不符合——我們并不知道需要多少對象以及如何構(gòu)造(比如構(gòu)造TCP的通訊socket需要accept()接受客戶端的信息),在這種情況下,我們只能在線程中去構(gòu)造對象,這樣的對象生命周期跟線程函數(shù)一樣。
            或者說,如果我們希望用類來封裝線程,那么這些可以調(diào)用線程的對象必須是全局的。相關(guān)內(nèi)容請參考本人前面的教程“初試多線程”等。
            国产精品日韩欧美久久综合| 国产精品99久久久久久董美香| 久久国产精品免费| 国产精品乱码久久久久久软件| 伊人久久国产免费观看视频| 久久人人添人人爽添人人片牛牛| 久久亚洲精品成人av无码网站| 国产国产成人精品久久| 亚洲国产一成久久精品国产成人综合| 91麻豆国产精品91久久久| 97久久精品无码一区二区| 午夜精品久久久久久| 久久九九有精品国产23百花影院| 久久精品国产一区二区电影| 久久久久免费看成人影片| 久久99精品久久久久久秒播| 久久久久免费看成人影片| 麻豆av久久av盛宴av| 久久本道综合久久伊人| 韩国免费A级毛片久久| 久久久久久久精品成人热色戒| 色综合久久中文色婷婷| 国产精品99精品久久免费| 香蕉久久夜色精品国产尤物| 国产成人精品久久综合 | 久久天堂AV综合合色蜜桃网 | 免费精品久久天干天干| 9191精品国产免费久久| 99久久中文字幕| 久久棈精品久久久久久噜噜| 少妇精品久久久一区二区三区| 亚洲Av无码国产情品久久| 久久久久噜噜噜亚洲熟女综合 | 99久久无码一区人妻| 精品综合久久久久久97超人| 韩国免费A级毛片久久| 亚洲国产另类久久久精品黑人| 国产A三级久久精品| 久久久久精品国产亚洲AV无码| 囯产精品久久久久久久久蜜桃| 久久精品国产亚洲AV香蕉|