• <ruby id="5koa6"></ruby>
    <ruby id="5koa6"><option id="5koa6"><thead id="5koa6"></thead></option></ruby>

    <progress id="5koa6"></progress>

  • <strong id="5koa6"></strong>
  • Visual C++中回調函數使用的變身大法

    發表于:2007-04-27來源:作者:點擊數: 標簽:中回用的變身C++visual
    對于回調函數的編寫始終是寫特殊處理功能程序時用到的技巧之一。先介紹一下回調的使用基本方法與原理。 1、在這里設:回調函數為A()(這是最簡單的情況,不帶參數,但我們應用的實際情況常常很會復雜),使用回調函數的操作函數為B(), 但B函數是需要參數的,
      對于回調函數的編寫始終是寫特殊處理功能程序時用到的技巧之一。先介紹一下回調的使用基本方法與原理。

      1、在這里設:回調函數為A()(這是最簡單的情況,不帶參數,但我們應用的實際情況常常很會復雜),使用回調函數的操作函數為B(), 但B函數是需要參數的,這個參數就是指向函數A的地址變量,這個變量一般就是函數指針。使用方法為:

    clearcase/" target="_blank" >cccccc width="90%" align=center bgColor=#e3e3e3 border=1>
    int A(char *p); // 回調函數
    typedef int(*CallBack)(char *p) ; // 聲明CallBack 類型的函數指針
    CallBack myCallBack ; // 聲明函數指針變量
    myCallBack = A; // 得到了函數A的地址

      B函數一般會寫為 B(CallBack lpCall,char * P,........); // 此處省略了p后的參數形式 。

      所以回調機制可解為,函數B要完成一定功能,但他自己是無法實現全部功能的。 需要借助于函數A來完成,也就是回調函數。B的實現為:

    B(CallBack lpCall,char *pProvide)
    {
     ........... // B 的自己實現功能語句
     lpCall(PpProvide); // 借助回調完成的功能 ,也就是A函數來處理的。
     ........... // B 的自己實現功能語句
    }
    // -------------- 使用例子 -------------
    char *p = "hello!";
    CallBack myCallBack ;
    myCallBack = A ;
    B(A, p);

      以上就是回調的基本應用,本文所說的變身,其實是利用傳入不同的函數地址,實現調用者類與回調函數所在類的不同轉換。

      1、問題描述

      CUploadFile 類完成數據上傳,與相應的界面進度顯示。

      主要函數Send(...) 和回調函數 GetCurState() ;

    class CUploadFile : public CDialog
    {
     ......
     int Send(LPCTSTR lpServerIP, LPCTSTR lpServerPort, LPCTSTR UploadFilePath) ;
     static int GetCurState(int nCurDone, int nInAll, void * pParam) ;
     ......
    }
    int CUploadFile ::Send(LPCTSTR lpServerIP, LPCTSTR lpServerPort, LPCTSTR UploadFilePath)
    {
     ... // 導出傳輸數據的函數
     int ret = Upload( (LPSTR)(LPCTSTR)m_strData,
        GetCurState, // 在這個回調函數中處理界面
        this, // CUploadFile 的自身指針 ,也就是pParam 所接受的參數
        (LPSTR)(LPCTSTR)UploadFilePath,
        "",
        "",
     );
    }
    int CUploadFile ::GetCurState(int nCurData, int nInAll, void * pParam)
    {
     .........
     UploadFile *pThis = (UploadFile *)pParam; // nCurData 當前以傳出的數據量
     // nInAll 總的數據量
     // 有了pThis可以對界面進行各種操作了。
     .............
    }

      但大家仔細觀察就可以發現,這個類把數據傳送和界面顯示聚和到了一起,不容易得到復用。而且在復用過程中需要改動較多的地方 。

      請大家記住現在的回調函數傳入的類本身的靜態成員函數。

      現在我們把數據的傳送和界面的顯示分離?;卣{則要傳入的是界面處理類的靜態函數。

      界面處理類 CShowGUI,數據上傳類 CUploadData

    class CUploadData
    {
     ......
     typedef int(*SetUploadCaller)(int nCurData, int nInAll, void * pParam);
     int UploadFile(LPCTSTR lpFileNamePath,LPVOID lparam,SetUploadCaller Caller );
     // 接受外界出入的參數,主要是回調函數的地址通過參數Caller,
     int Send(LPCTSTR lpServerIP, LPCTSTR lpServerPort, LPCTSTR UploadFilePath) ;
     ...... // 注意此時不在需要GetCurState 函數了 。
    }

    class CShowGUI: public CDialog
    {
     .......
     typedef int(*SetUploadCaller)(int nCurData, int nInAll, void * pParam);
     void SetCallBack(LPCTSTR strPath);
     static int GetCurState(int nCurData, int nInAll, void * pParam) ;
     CUploadData m_Uploa
     d ; // 數據上傳類是界面顯示類的一個成員變量。
     .......
    }

    void CShowGUI :: SetCallBack(LPCTSTR strPath)
    {
     CUploadData myUploadData ;
     SetUploadCaller myCaller; // 聲明一個函數指針變量
     myCaller = CurState ; // 取得界面處理函數的地址
     myUploadData .UploadFile(strPath,this,myCaller); // 界面處理類的函數傳入,實現了數據傳入與界面處理的分離 .
    }

      通過上面的演示做到了界面與數據的分離,回調函數分別扮演了不同角色,所以隨著處理問題的不同應靈活應用,但同樣因為處理數據類不知道界面處理類或外部調用類的類型,而更無法靈活地處理界面的不同顯示方式。這方面還希望喜歡鉆研技術的朋友繼續研究。

    原文轉自:http://www.kjueaiud.com

    評論列表(網友評論僅供網友表達個人看法,并不表明本站同意其觀點或證實其描述)
    ...
    老湿亚洲永久精品ww47香蕉图片_日韩欧美中文字幕北美法律_国产AV永久无码天堂影院_久久婷婷综合色丁香五月

  • <ruby id="5koa6"></ruby>
    <ruby id="5koa6"><option id="5koa6"><thead id="5koa6"></thead></option></ruby>

    <progress id="5koa6"></progress>

  • <strong id="5koa6"></strong>