類成員函數可以為回調函數嗎


    關於類成員函數是否可以成為回調函數,我們首先需要明確幾個定義,1. 什么是回調函數 2. 為什么要使用回調函數 3. 調用普通類成員函數和調用回調函數有什么區別

    什么是回調函數?

        簡而言之,回調函數就是一個通過函數指針調用的函數,如果你把函數的指針(地址)作為參數傳遞給另一個函數,當這個指針被用為調用它所指向的函數時,我們就說這是回調函數

    為什么要使用回調函數?

  因為可以把調用者與被調用者分開。調用者不關心誰是被調用者,所有它需知道的,只是存在一個具有某種特定原型、某些限制條件(如返回值為int)的被調用函數。

    對普通函數的調用:調用程序發出對普通函數的調用后,程序執行立即轉向被調用函數執行,直到被調用函數執行完畢后,再返回調用程序繼續執行。從發出調用的程序的角度來看,這個過程為“調用-->等待被調用函數執行完畢-->繼續執行”

    對回調函數的調用:調用程序發出對回調函數的調用后,不等函數執行完畢,立即返回並繼續執行。這樣,調用程序執行和被調用函數同時在執行。當被調函數執行完畢后,被調函數會反過來調用某個事先指定函數,以通知調用程序:函數調用結束。這個過程稱為回調(CallBack),這正是回調函數名稱的由來

 

    如何成為回調函數?

        1. 原理:

            如果試圖直接使用C++的成員函數作為回調函數將發生錯誤,甚至編譯就不能通過。 其錯誤原因是普通的C++成員函數都隱含了一個傳遞函數作為參數,亦即“this”指針,C++通過傳遞this指針給其成員函數從而實現程序函數可以訪問C++的數據成員。這也可以理解為什么C++類的多個實例可以共享成員函數卻-有不同的數據成員。由於this指針的作用,使得將一個CALL-BACK型的成員函數作為回調函數安裝時就會因為隱含的this指針使得函數參數個數不匹配,從而導致回調函數安裝失敗。

        2. 解決方案

            要解決這一問題的關鍵就是不讓this指針起作用,通過采用以下兩種典型技術可以解決在C++中使用回調函數所遇到的問題。這種方法具有通用性,適合於任何C++。   

       1).   不使用成員函數,為了訪問類的成員變量,可以使用友元操作符(friend),在C++中將該函數說明為類的友元即可。         

            2).   使用靜態成員函數,靜態成員函數不使用this指針作為隱含參數,這樣就可以作為回調函數了。靜態成員函數具有兩大特點:其一,可以在沒有類實例的情況下使用;其二,只能訪問靜態成員變量和靜態成員函數,不能訪問非靜態成員變量和非靜態成員函數。由於在C++中使用類成員函數作為回調函數的目的就是為了訪問所有的成員變量和成員函數,如果做不到這一點將不具有實際意義。解決的辦法也很簡單,就是使用一個靜態類指針作為類成員,通過在類創建時初始化該靜態指針,如pThis=this,然后在回調函數中通過該靜態指針就可以訪問所有成員變量和成員函數了。這種處理辦法適用於只有一個類實例的情況,因為多個類實例將共享靜態類成員和靜態成員函數,這就導致靜態指針指向最后創建的類實例。為了避免這種情況,可以使用回調函數的一個參數來傳遞this指針,從而實現數據成員共享。這種方法稍稍麻煩,這里就不再贅述。

  回調函數實例

    舉個生活中常見的例子:你在網上買了東西通過快遞來配送,給你兩個選擇:

  1. 你可以一遍遍的打快遞公司的電話,查詢你的貨是否到了,是否可以領取;

  2.你可以安心的干自己的事,等配送人員把貨送到你家門口打電話通知你,你去領取;

你會選擇哪種方式?當然你會選擇第二種,好處不言而喻,這也正是回調函數的形象解釋。

  

 1 #include<iostream>
 2 using namespace std;
 3 
 4 void Sum(int a, int b) 
 5 {
 6     int c = a + b;
 7     cout << "Sum = "<< c <<endl;
 8 }
 9 typedef void (*FuncCallBack)(int, int);
10 
11 void GetCallBack(const int i, FuncCallBack cb)
12 {
13     if (1 == i)
14     {
15         cb(3, 4);
16     }
17 }
18 
19 void main()
20 {
21     GetCallBack(1, &Sum);
22 }

 

  運行結果:Sum = 7;對於上面這個例子FuncCallBack是回調函數的定義,Sum是其實現,

  在SwcDefine,SwcKey和SwcManager,中有例子可以對比


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM