• <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>

            C++ Programmer

            天行健,君子以自強不息; 地勢坤,君子以厚德載物

            多線程編程淺析(2)——線程間通信

               上文我們介紹了如何建立一個簡單的多線程程序,多線程之間不可避免的需要進行通信。相比于進程間通信來說,線程間通信無疑是相對比較簡單的。

               首先我們來看看最簡單的方法,那就是使用全局變量(靜態(tài)變量也可以)來進行通信,由于屬于同一個進程的各個線程是處于同一個進程空間中的,并且它們共享這個進程的各種資源,因此它們都可以毫無障礙的訪問這個進程中的全局變量。當需要有多個線程來訪問一個全局變量時,通常我們會在這個全局變量前加上volatile聲明,來告訴編譯器這個全局變量是易變的,讓編譯器不要對這個變量進行優(yōu)化(至于編譯器到底有沒有按照你的要求來對volatile進行處理這個暫且不理)。

               下面貼出一段簡單的示例代碼:

            #include "stdafx.h"
            #include 
            "windows.h"
            #include 
            "stdio.h"

            volatile int ThreadData = 0;

            void ThreadProcess()
            {
                
            for(int i=0; i<6; i++)
                
            {
                    ThreadData 
            += 1000;
                    Sleep(
            1000);
                    printf(
            "Sub  Thread Tick %5d! %5d\n",(i+1)*1000, ThreadData);
                }

                printf(
            "Exit Sub Thread!\n");
                
            }


            int _tmain(int argc, _TCHAR* argv[])
            {
                HANDLE hThread;
                DWORD ThreadID;
                hThread
            =CreateThread(NULL,
                                 
            0,
                                 (LPTHREAD_START_ROUTINE)ThreadProcess,
                                 NULL,
                                 
            0,
                                 
            &ThreadID);
                
                
            for(int i=0; i<10; i++)
                
            {
                    ThreadData 
            -= 600;
                    Sleep(
            600);
                    printf(
            "Main Thread Tick %5d! %5d\n", (i+1)*600, ThreadData);
                }

                printf(
            "Main Thread Loop Finished! \n");
                system(
            "pause");
                
            return 0;
            }

               除了全局變量之外,還有其他的方法,比如利用消息機制等來實現(xiàn)線程間通信。這個就不詳細解釋了,關于消息機制,詳情請看Windows消息機制概述 。 

               下面,關于多線程中的全局變量,我來介紹點有點偏題的東西:
            線程局部存儲(TLS)
                進程中的全局變量與函數(shù)內定義的靜態(tài)(static)變量,是各個線程都可以訪問的共享變量。在一個線程修改的內存內容,對所有線程都生效。這是一個優(yōu)點也是一個缺點。說它是優(yōu)點,線程的數(shù)據(jù)交換變得非??旖荨Uf它是缺點,一個線程死掉了,其它線程也性命不保; 多個線程訪問共享數(shù)據(jù),需要昂貴的同步開銷,也容易造成同步相關的BUG。
              如果需要在一個線程內部的各個函數(shù)調用都能訪問、但其它線程不能訪問的變量(被稱為static memory local to a thread 線程局部靜態(tài)變量),就需要新的機制來實現(xiàn)。這就是TLS。
              線程局部存儲在不同的平臺有不同的實現(xiàn),可移植性不太好。 
              方法一:每個線程創(chuàng)建時系統(tǒng)給它分配一個LPVOID指針的數(shù)組(叫做TLS數(shù)組),這個數(shù)組從C編程角度是隱藏著的不能直接訪問,需要通過一些C API函數(shù)調用訪問。首先定義一些DWORD線程全局變量或函數(shù)靜態(tài)變量,準備作為各個線程訪問自己的TLS數(shù)組的索引變量。一個線程使用TLS時,第一步在線程內調用TlsAlloc()函數(shù),為一個TLS數(shù)組索引變量與這個線程的TLS數(shù)組的某個槽(slot)關聯(lián)起來,例如獲得一個索引變量:
              global_dwTLSindex=TLSAlloc();
              注意,此步之后,當前線程實際上訪問的是這個TLS數(shù)組索引變量的線程內的拷貝版本。也就說,不同線程雖然看起來用的是同名的TLS數(shù)組索引變量,但實際上各個線程得到的可能是不同DWORD值。其意義在于,每個使用TLS的線程獲得了一個DWORD類型的線程局部靜態(tài)變量作為TLS數(shù)組的索引變量。C/C++原本沒有直接定義線程局部靜態(tài)變量的機制,所以在如此大費周折。
              第二步,為當前線程動態(tài)分配一塊內存區(qū)域(使用LocalAlloc()函數(shù)調用),然后把指向這塊內存區(qū)域的指針放入TLS數(shù)組相應的槽中(使用TlsValue()函數(shù)調用)。
              第三步,在當前線程的任何函數(shù)內,都可以通過TLS數(shù)組的索引變量,使用TlsGetValue()函數(shù)得到上一步的那塊內存區(qū)域的指針,然后就可以進行內存區(qū)域的讀寫操作了。這就實現(xiàn)了在一個線程內部這個范圍處處可訪問的變量。
              最后,如果不再需要上述線程局部靜態(tài)變量,要動態(tài)釋放掉這塊內存區(qū)域(使用LocalFree()函數(shù)),然后從TLS數(shù)組中放棄對應的槽(使用TlsFree()函數(shù))。
              方法二:
              直接聲明這個變量是各個線程有自己拷貝的線程局部靜態(tài)變量:
              __declspec( thread ) int var_name;

            posted on 2009-07-23 17:18 Saga 閱讀(15766) 評論(1)  編輯 收藏 引用 所屬分類: MultiTask

            評論

            # re: 多線程編程淺析(2)——線程間通信 2009-07-25 15:56 12530彩鈴

            來學習哦~  回復  更多評論   

            導航

            <2009年7月>
            2829301234
            567891011
            12131415161718
            19202122232425
            2627282930311
            2345678

            統(tǒng)計

            常用鏈接

            留言簿(1)

            隨筆分類

            隨筆檔案

            搜索

            積分與排名

            最新評論

            閱讀排行榜

            評論排行榜

            久久棈精品久久久久久噜噜| 久久水蜜桃亚洲av无码精品麻豆| 九九久久自然熟的香蕉图片| 香蕉久久夜色精品国产2020| 99久久精品免费| 国产精品一久久香蕉国产线看观看| 亚洲欧美成人久久综合中文网| 国产精品成人久久久久久久| 精品久久久噜噜噜久久久| 日产精品久久久久久久| 精品人妻伦九区久久AAA片69| 久久婷婷午色综合夜啪| 日韩久久无码免费毛片软件| 精品久久国产一区二区三区香蕉| 精品免费久久久久国产一区 | 精品久久久久久国产91| 麻豆一区二区99久久久久| 久久免费的精品国产V∧| 996久久国产精品线观看| 99久久综合狠狠综合久久止| 久久这里只有精品首页| 99久久精品国产一区二区| 久久精品国产色蜜蜜麻豆 | 久久青青国产| 四虎国产精品成人免费久久| 97香蕉久久夜色精品国产| 囯产精品久久久久久久久蜜桃| 久久精品国产亚洲AV香蕉| 伊人久久久AV老熟妇色| 77777亚洲午夜久久多喷| 99热热久久这里只有精品68| 欧美精品福利视频一区二区三区久久久精品 | 久久国产精品免费一区| 午夜精品久久久内射近拍高清| 伊人久久综合成人网| 色综合久久天天综合| 欧美与黑人午夜性猛交久久久| 色天使久久综合网天天| 99re久久精品国产首页2020| 精品99久久aaa一级毛片| 亚洲av日韩精品久久久久久a |