• <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>
            幽幽
             
            posts - 51,  comments - 28,  trackbacks - 0

            我以前寫線程時要么老老實實照著聲明寫,要么使用C++類的靜態成員函數來作為回調函數,經常會因為線程代碼而破壞封裝.之前雖然知道類成員函數的展開形式,但從沒想過利用過它,昨天看深入ATL時無意中學會了這一招:) 

            類成員方法是一個比較特殊的函數,它在編譯時會被轉化成普通函數,比如有TMyClass類:
            class TMyClass{
                void Func();
            };

            這個TMyClass::Func最終會轉化成 void Func(TMyClass *this); 也就是說在原第一個參數前插入指向對象本身的this指針。

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

            現在做個實驗:
            #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++)       // 根據m_nCount成員打印一排數字
                    {
                        printf("Class%d : %d\n",m_nId,i);
                    }
                }
            };

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

                TMyClass MyClass1(1,10),MyClass2(2,5); // 產生兩個TMyClass對象

                Proc.MemberProc = &TMyClass::ThreadProc;   // 轉換,Proc.ThreadProc就是對應的普通函數指針了

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

            運行!神奇吧?:-)

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

            轉自:http://blog.csdn.net/waiting4you/archive/2007/12/29/2000796.aspx

            posted on 2008-07-22 03:13 幽幽 閱讀(3232) 評論(1)  編輯 收藏 引用 所屬分類: Windows

            FeedBack:
            # re: 把C++類成員方法直接作為線程回調函數
            2009-06-07 01:56 | JAXE
            老大,你博客的音樂把我嚇了一大跳啊。。。我半夜聽到音樂開頭,很詭異的感覺啊,當時沒反應過來時你的博客在放。。  回復  更多評論
              

            <2008年7月>
            293012345
            6789101112
            13141516171819
            20212223242526
            272829303112
            3456789

            常用鏈接

            留言簿(6)

            隨筆分類(35)

            隨筆檔案(51)

            文章分類(3)

            文章檔案(3)

            相冊

            我的鏈接

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            精品久久久无码21p发布| 久久综合狠狠综合久久综合88| 女人高潮久久久叫人喷水| 国产69精品久久久久99| 狠狠色丁香久久综合婷婷| 久久97精品久久久久久久不卡| 国产91色综合久久免费| 国产成人精品久久| 久久天天躁夜夜躁狠狠| 日韩人妻无码精品久久久不卡 | 久久精品国产亚洲Aⅴ香蕉| 色综合久久最新中文字幕| 久久99精品久久久久久水蜜桃| 九九久久99综合一区二区| 精品久久久久久国产三级| 手机看片久久高清国产日韩| 99久久人人爽亚洲精品美女| 精品久久久久久久| 狠狠精品久久久无码中文字幕| 少妇精品久久久一区二区三区| 久久99毛片免费观看不卡| 狠狠色伊人久久精品综合网| 精品无码人妻久久久久久| 欧美日韩精品久久免费| 99久久人妻无码精品系列蜜桃| 中文字幕亚洲综合久久2| 国产一区二区久久久| 9191精品国产免费久久| 久久久亚洲AV波多野结衣| 亚洲国产精品人久久| 精品久久久久久中文字幕大豆网 | 久久丝袜精品中文字幕| 精品久久亚洲中文无码| 久久亚洲AV无码西西人体| 久久亚洲美女精品国产精品| 香蕉久久一区二区不卡无毒影院 | 久久久久久久综合狠狠综合| 亚洲中文字幕无码久久综合网 | 久久香蕉一级毛片| 狠狠色婷婷久久一区二区三区| 久久综合偷偷噜噜噜色|