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

            woaidongmao

            文章均收錄自他人博客,但不喜標(biāo)題前加-[轉(zhuǎn)貼],因其丑陋,見(jiàn)諒!~
            隨筆 - 1469, 文章 - 0, 評(píng)論 - 661, 引用 - 0
            數(shù)據(jù)加載中……

            把C++類成員方法直接作為線程回調(diào)函數(shù)

            我以前寫(xiě)線程時(shí)要么老老實(shí)實(shí)照著聲明寫(xiě),要么使用C++類的靜態(tài)成員函數(shù)來(lái)作為回調(diào)函數(shù),經(jīng)常會(huì)因?yàn)榫€程代碼而破壞封裝.之前雖然知道類成員函數(shù)的展開(kāi)形式,但從沒(méi)想過(guò)利用過(guò)它,昨天看深入ATL時(shí)無(wú)意中學(xué)會(huì)了這一招:)

            類成員方法是一個(gè)比較特殊的函數(shù),它在編譯時(shí)會(huì)被轉(zhuǎn)化成普通函數(shù),比如有TMyClass:
            class TMyClass{
                void Func();
            };

            這個(gè)TMyClass::Func最終會(huì)轉(zhuǎn)化成 void Func(TMyClass *this); 也就是說(shuō)在原第一個(gè)參數(shù)前插入指向?qū)ο蟊旧淼?span lang="EN-US">this指針。

            我們可以利用這個(gè)特性寫(xiě)一個(gè)非靜態(tài)類成員方法來(lái)直接作為線程回調(diào)函數(shù),先看_beginthread函數(shù)的定義:
            unsigned long _RTLENTRY _EXPFUNC _beginthread (void (_USERENTRY *__start)(void *),unsigned __stksize, void *__arg);
            其中的第一個(gè)參數(shù)就是作為線程執(zhí)行主體的回調(diào)函數(shù)。它的原型是:void Func(void *),這個(gè)void*參數(shù)是作為自定義數(shù)據(jù)傳入的。對(duì)比一下上面所說(shuō)的TMyClass::Func的最終形式,它正好可以符合這里的要求。

            現(xiàn)在做個(gè)實(shí)驗(yàn):
            #include <stdio.h>
            #include <process.h>

            class TMyClass{
                int m_nCount;
                int m_nId;
            public:
                TMyClass(int nId,int nCount)
                    :m_nId(nId),m_nCount(nCount)
                {
                }

                void _USERENTRY ThreadProc()            // 類成員方法
                {
                    for(int i=0; i<m_nCount; i++)       //
            根據(jù)m_nCount成員打印一排數(shù)字
                    {
                        printf("Class%d : %d\n",m_nId,i);
                    }
                }
            };

            int main(int argc, char* argv[])
            {
               
            union {                                // 聯(lián)合類,用于轉(zhuǎn)換類成員方法指針到普通函數(shù)指針(試過(guò)編譯器不允許在這兩種函數(shù)之間強(qiáng)制轉(zhuǎn)換),不知道有沒(méi)有更好的方法。
                    void (_USERENTRY *ThreadProc)(void *);
                    void (_USERENTRY TMyClass::*MemberProc)();
                } Proc;  
                                         // 盡管聯(lián)合里的兩種函數(shù)類型現(xiàn)在看起來(lái)有很大不同,但它們的最終形式是相同的。

                TMyClass MyClass1(1,10),MyClass2(2,5); // 產(chǎn)生兩個(gè)TMyClass對(duì)象

                Proc.MemberProc = &TMyClass::ThreadProc;   // 轉(zhuǎn)換,Proc.ThreadProc就是對(duì)應(yīng)的普通函數(shù)指針了

                _beginthread(Proc.ThreadProc,4096,&MyClass1);   // 開(kāi)始線程,這里的Proc.ThreadProc實(shí)際上是TMyClass::ThreadProc, 它要的this指針是我們給的&MyClass1
                _beginthread(Proc.ThreadProc,4096,&MyClass2);
                system("pause");
                return 0;
            }

            運(yùn)行!神奇吧?:-)

            其實(shí)不止線程回調(diào)函數(shù),其實(shí)只要是形如Func(void*,...)的回調(diào)函數(shù)都可以用這種方法直接使用類成員方法。(前提是第一個(gè)void*是自定義數(shù)據(jù),也就是說(shuō)它不能有其它功能)

             

            posted on 2008-11-09 18:42 肥仔 閱讀(654) 評(píng)論(0)  編輯 收藏 引用 所屬分類: C++ 基礎(chǔ)

            久久久久久久综合日本亚洲 | 中文字幕久久波多野结衣av| 国内精品久久久久久久亚洲| 久久久人妻精品无码一区| 久久久精品人妻无码专区不卡| 欧美伊人久久大香线蕉综合69| 欧美黑人又粗又大久久久| 精品久久久久久无码中文字幕 | 久久久亚洲欧洲日产国码aⅴ | 久久91精品综合国产首页| 久久综合偷偷噜噜噜色| 99久久无色码中文字幕 | 97超级碰碰碰久久久久| 亚洲色欲久久久久综合网 | 91麻精品国产91久久久久| 伊人久久大香线蕉综合Av| 国产毛片久久久久久国产毛片| 久久无码人妻一区二区三区午夜| 无码任你躁久久久久久老妇| 久久国产精品久久精品国产| 久久人做人爽一区二区三区| 久久精品无码一区二区三区日韩| 久久成人国产精品二三区| 久久精品国产清高在天天线| 久久精品国产亚洲AV久| 伊人久久成人成综合网222| 日韩精品久久久久久| 国产精品美女久久久久网| 99久久免费国产精品热| 久久精品国产亚洲77777| 久久国产亚洲高清观看| 久久人人爽人人爽人人AV东京热| 国产激情久久久久久熟女老人| 亚洲日本va午夜中文字幕久久| 伊人久久成人成综合网222| 无码精品久久一区二区三区 | 亚洲一区中文字幕久久| 国产精品美女久久久免费| 91精品国产91久久久久久青草| 日本精品久久久久中文字幕| 国产激情久久久久影院老熟女免费|