• <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>
            隨筆 - 505  文章 - 1034  trackbacks - 0
            <2007年8月>
            2930311234
            567891011
            12131415161718
            19202122232425
            2627282930311
            2345678


            子曾經(jīng)曰過:編程無他,唯手熟爾!

            常用鏈接

            留言簿(94)

            隨筆分類(649)

            隨筆檔案(505)

            相冊

            BCB

            Crytek

            • crymod
            • Crytek's Offical Modding Portal

            Game Industry

            OGRE

            other

            Programmers

            Qt

            WOW Stuff

            搜索

            •  

            積分與排名

            • 積分 - 912010
            • 排名 - 14

            最新隨筆

            最新評論

            閱讀排行榜

            評論排行榜

            阿來點評:這篇文章寫得真是好,無論是內(nèi)容還是排版,我一直有把member function 作為回調(diào)函數(shù)的欲望,但是試了一次發(fā)現(xiàn)不行;現(xiàn)在知道是隱含的this指針的問題。總之,這篇文章不僅解開了我心中的疑團,還提供了兩種解決方案,我真是感激地痛苦流涕啊!



            提出問題:

            回調(diào)函數(shù)是基于C編程的Windows SDK的技術,不是針對C++的,程序員可以將一個C函數(shù)直接作為回調(diào)函數(shù),但是如果試圖直接使用C++的成員函數(shù)作為回調(diào)函數(shù)將發(fā)生錯誤,甚至編譯就不能通過。
            分析原因:
            普通的C++成員函數(shù)都隱含了一個傳遞函數(shù)作為參數(shù),亦即“this”指針,C++通過傳遞一個指向自身的指針給其成員函數(shù)從而實現(xiàn)程序函數(shù)可以訪問C++的數(shù)據(jù)成員。這也可以理解為什么C++類的多個實例可以共享成員函數(shù)但是確有不同的數(shù)據(jù)成員。由于this指針的作用,使得將一個CALLBACK型的成員函數(shù)作為回調(diào)函數(shù)安裝時就會因為隱含的this指針使得函數(shù)參數(shù)個數(shù)不匹配,從而導致回調(diào)函數(shù)安裝失敗
            解決方案:
            一,不使用成員函數(shù),直接使用普通C函數(shù),為了實現(xiàn)在C函數(shù)中可以訪問類的成員變量,可以使用友元操作符(friend),在C++中將該C函數(shù)說明為類的友元即可。這種處理機制與普通的C編程中使用回調(diào)函數(shù)一樣。
            二,使用靜態(tài)成員函數(shù),靜態(tài)成員函數(shù)不使用this指針作為隱含參數(shù),這樣就可以作為回調(diào)函數(shù)了。靜態(tài)成員函數(shù)具有兩大特點:其一,可以在沒有類實例的情況下使用;其二,只能訪問靜態(tài)成員變量和靜態(tài)成員函數(shù),不能訪問非靜態(tài)成員變量和非靜態(tài)成員函數(shù)。由于在C++中使用類成員函數(shù)作為回調(diào)函數(shù)的目的就是為了訪問所有的成員變量和成員函數(shù),如果作不到這一點將不具有實際意義。我們通過使用靜態(tài)成員函數(shù)對非靜態(tài)成員函數(shù)包裝的辦法來解決問題。類實例可以通過附加參數(shù)全局變量的方式的方式傳遞到靜態(tài)成員函數(shù)中。分別舉例如下:
            1,參數(shù)傳遞的方式
               #include <iostream.h>  
               class TClassA
               {
               public:

             

                  void Display(const char* text) { cout << text << endl; };
                  static void Wrapper_To_Call_Display(void* pt2Object, char* text);
                  // more....
               };

               // 靜態(tài)包裝函數(shù),能夠調(diào)用成員函數(shù)Display(),本身做為回調(diào)函數(shù)來使用
               void TClassA::Wrapper_To_Call_Display(void* pt2Object, char* string)
               {
                   // 顯式類型轉換
                   TClassA* mySelf = (TClassA*) pt2Object;

                   // 調(diào)用普通成員函數(shù)
                   mySelf->Display(string);
               }

               // 回調(diào)函數(shù)的宿主,在這里回調(diào)用函數(shù)被使用
               void DoItA(void* pt2Object, void (*pt2Function)(void* pt2Object, char* text))
               {
                  // 使用回調(diào)函數(shù)
                  pt2Function(pt2Object, "hi, i'm calling back using a argument ;-)"); 
               }

               // 執(zhí)行示例
               void Callback_Using_Argument()
               {
                  TClassA objA;
                  DoItA((void*) &objA, TClassA::Wrapper_To_Call_Display);
               }

            2,全局變量的方式
               #include <iostream.h>  
               void* pt2Object;        // 全局變量,可以指向任意對象
               class TClassB
               {
               public:

                  void Display(const char* text) { cout << text << endl; };
                  static void Wrapper_To_Call_Display(char* text);

               };

               // 靜態(tài)的包裝函數(shù)
               void TClassB::Wrapper_To_Call_Display(char* string)
               {
                   //需要保證全局變量值的正確性
                   TClassB* mySelf = (TClassB*) pt2Object;
                   mySelf->Display(string);
               }

               // 回調(diào)用函數(shù)的宿主,在這里回調(diào)用函數(shù)被使用
               void DoItB(void (*pt2Function)(char* text))
               {
              
                  pt2Function("hi, i'm calling back using a global ;-)");   // make callback
               }

               // 執(zhí)行示例
               void Callback_Using_Global()
               {
                  TClassB objB;  
                  pt2Object = (void*) &objB;
                  DoItB(TClassB::Wrapper_To_Call_Display);
               }

            注意:通過上面兩種方法的比較可以看出,第2種方法中靜態(tài)包裝函數(shù)可以和普通成員函數(shù)保持簽名一致,當回調(diào)函數(shù)的宿主接口不能改變時,這種方法特別有用。但因為使用了全局變量,也不是一個好的設計。

            posted on 2007-08-06 10:21 七星重劍 閱讀(1062) 評論(3)  編輯 收藏 引用 所屬分類: PL--c/c++

            FeedBack:
            # re: [轉]將類成員函數(shù)用做C回調(diào)函數(shù) 2007-11-12 14:18 stepchen87
            你寫的例子太亂了,一個簡單的問題代碼體現(xiàn)得這么混亂。  回復  更多評論
              
            # re: [轉]將類成員函數(shù)用做C回調(diào)函數(shù) 2007-11-13 06:58 重劍
            @stepchen87
            這個代碼一點都不亂啊,才幾行啊
            小伙兒耐心點啊
            要不你寫幾行簡潔的?  回復  更多評論
              
            久久久久亚洲AV无码观看| 成人久久综合网| 亚洲精品乱码久久久久久不卡| 久久精品成人免费国产片小草| 久久97久久97精品免视看| 久久香综合精品久久伊人| 国产一级做a爰片久久毛片| 99久久精品免费观看国产| 欧美性大战久久久久久| 久久人爽人人爽人人片AV | 超级碰久久免费公开视频| 国产精品久久久久乳精品爆| 色妞色综合久久夜夜| 久久亚洲高清观看| 亚洲国产精品久久电影欧美| 岛国搬运www久久| 国内精品久久久人妻中文字幕| 久久久久久久综合综合狠狠| 国内精品久久久久久99蜜桃| A级毛片无码久久精品免费| 久久人人爽人人爽人人片AV不 | www.久久99| 久久99热这里只有精品国产| 久久精品国产福利国产琪琪| 久久ZYZ资源站无码中文动漫| 亚洲人成无码网站久久99热国产| 1000部精品久久久久久久久| 中文国产成人精品久久亚洲精品AⅤ无码精品| 久久国产精品-久久精品| 熟妇人妻久久中文字幕| 色欲综合久久躁天天躁蜜桃| 久久国产亚洲精品| 久久久久香蕉视频| 久久av高潮av无码av喷吹| 国产一区二区三区久久| 久久久久久夜精品精品免费啦| 国内高清久久久久久| 国产69精品久久久久9999APGF| 国产精品久久新婚兰兰| 亚洲精品国产字幕久久不卡| 国产精品亚洲综合久久|