C++中的類成員函數(shù)當(dāng)線程函數(shù)
類成員函數(shù)當(dāng)線程函數(shù)
C++類成員函數(shù)使用時,都會隱式傳遞一個this指針給該函數(shù),this指針指向該類的對象。函數(shù)體可以通過顯示調(diào)用該指針或直接訪問類內(nèi)成員。
回調(diào)函數(shù)是通過指針調(diào)用的函數(shù),最常使用的回調(diào)函數(shù)就是在創(chuàng)建線程時,以一個函數(shù)指針以及傳遞給這個函數(shù)多個參數(shù)來調(diào)用線程函數(shù)來創(chuàng)建線程。
那么一般的類成員函數(shù)是不能用作回調(diào)函數(shù)的,因為庫函數(shù)在使用回調(diào)函數(shù)時,都會傳遞指定的符合回調(diào)函數(shù)聲明的的參數(shù)給回調(diào)函數(shù),而類成員函數(shù)隱式包含一個this指針參數(shù),所以把類成員函數(shù)當(dāng)作回調(diào)函數(shù)編譯時因為參數(shù)不匹配會出錯。
std::thread,它的第一個參數(shù)為函數(shù)指針,在c++中這樣是獲取不到其成員函數(shù)的指針,所以會報錯。
解決方法一
把成員函數(shù)設(shè)成靜態(tài)成員函數(shù),不屬于某個對象,屬于整個類,沒有this指針。但是靜態(tài)成員函數(shù)并不能使用非靜態(tài)的成員變量(因為它沒有某個具體對象的this指針),可通過對象或者類指針調(diào)用。
解決方法二
把成員函數(shù)聲明為友元函數(shù),沒有this指針,但是能夠訪問類的成員變量。
解決方法三
假設(shè)需要在單獨的線程中調(diào)用類Hack的非靜態(tài)成員函數(shù)func2()。不用直接傳遞成員函數(shù)的地址到thr_create(),聲明一個帶 void* 參數(shù)的普通函數(shù) intermediary(void*),然后調(diào)用它:
void intermediary(void);
接著創(chuàng)建一個結(jié)構(gòu),結(jié)構(gòu)定義如下:
struct A { Hack * p; //類對象指針 void (Hack::*pmf)(); // 成員函數(shù)指針 };
創(chuàng)建一個結(jié)構(gòu)實例,用希望的對象地址和成員函數(shù)地址填充結(jié)構(gòu):
A a; // 結(jié)構(gòu)實例 Hack h; // 創(chuàng)建對象 //填充結(jié)構(gòu) a.p = & h; a.pmf = &Hack::func2; // 取成員函數(shù)地址
現(xiàn)在回過頭來實現(xiàn)intermediary()函數(shù):
void *intermediary(void* ptr) { A* pa=static_cast < A* > (ptr); // 強制轉(zhuǎn)換 p 為 A* Hack* ph=pa->p; // 從A中析取Hack對象地址 void (Hack::*pmf)()=pa->pmf; // 析取 ptr 到成員函數(shù) (ph->*pmf)(); // 調(diào)用成員函數(shù) }
最后將intermediary()的地址傳遞到thr_create():
pthread_create (&ptid, NULL, intermediary, (void *)&a );
pthread_create()調(diào)用函數(shù)intermediary()并將A的地址傳遞給它。intermediary()再從其指針參數(shù)中展開結(jié)構(gòu)A并調(diào)用希望的成員函數(shù)。
這種間接方式的處理可以安全地在單獨線程中啟動成員函數(shù),即便是線程庫不支持成員函數(shù)。
如果需要調(diào)用不同類的不同成員函數(shù),可以將結(jié)構(gòu)A轉(zhuǎn)換成類模板,將函數(shù)intermediary()轉(zhuǎn)換成函數(shù)模板。從而編譯器便會自動產(chǎn)生大多數(shù)樣板文件代碼。
類成員函數(shù)作為多線程的入口
搜了一圈答案,基本上都是啟動線程的時候傳入this指針,在線程函數(shù)內(nèi)部再強轉(zhuǎn)的解決方案??赡茱@得有些別扭。
編譯器不允許強制轉(zhuǎn)換,那就用union來實現(xiàn)。
union { ? void *(*trfunc)(void *); ? void *(lock_client_cache::*memfunc)(); } func; ? func.memfunc = &lock_client_cache::do_thread; pthread_t pid; pthread_create(pid, 0, func.trfunc, this); pthread_detach(pid);
do_thread是非靜態(tài)類成員函數(shù),沒有參數(shù)。posix庫的情況下返回一個void*,win32的線程的情況下返回void。
*該方法適用于只需要傳遞this指針的情況,如果需要傳遞多個參數(shù),還要按老方法。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
C語言中strlen() strcpy() strcat() strcmp()函數(shù)的實現(xiàn)方法
這篇文章主要介紹了C語言中strlen() strcpy() strcat() strcmp()函數(shù)的實現(xiàn)方法,需要的朋友可以參考下2017-08-08c++標(biāo)準(zhǔn)輸入輸出流關(guān)系的前世今生
這篇文章主要給大家介紹了關(guān)于c++標(biāo)準(zhǔn)輸入輸出流關(guān)系的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04