看板 C_and_CPP 關於我們 聯絡資訊
※ 引述《sighAll (sigh)》之銘言: : 大家好 : 我有一個header, test.h, 有一個function pointer : typedef int(*pfnTest_t)(void* x, unsigned char* y, unsigned : int z); : 一個API: : int test_api(pfnTest_t p_pfnTest); : 然後有一個cpp : class B { : ... : test(); : ... : } : B::B_API(void* x, unsigned char* y, unsigned int z){ : ... : } : B::test() : { : test_api(&B::B_API); // error! : } : compiler的時候 有error : 請問這cpp裡面如何把function pointer帶進去?? : 謝謝! 既然你說了 test_api() 是純 C 寫的 lib 你不能動, 那就不要去想這個 test_api() 可以 invoke 你的成員函式指標。 除非你能把 test_api 改成這樣: int test_api(B *obj, B::B_API *ptr) { obj->*ptr(x, y, z); } 然後你的 B::test() 要改寫成這樣才行: B::test() { test_api(this, &B::B_API); } 非靜態成員函式指標不是函式指標,你要 invoke 它必須綁定一個物件。 C++ 提供了 .* 和 ->* 兩個特殊的運算子讓你執行這件事,這是 C++ 才有的東西。 詳情可以看我 11 年前寫的東西:#1AdlQLYS 然而你說 test_api() 不是你能動的,而且還位在純 C 的函式庫, 這麼一來無論上面這種修改方式就必須再繞個彎, 要把 obj->*ptr(x, y, z); 這行抽到另一個能由 pfnTest_t 指向的函式才能達成, 這樣你把那個函式的指標傳給 test_api() 就沒有問題了。 只不過你很快又會陷入另一個苦惱,就是如何保存 obj 和 ptr 讓那個函式能正確呼叫。 這個問題自古以來 C++ 要 call C API 就一直存在,也有很簡單粗暴的做法來解決。 當然如果今天你願意對 test_api() 做一些小修改,事情就能簡單很多。 以一個純 C 函式庫而言,少提供一個 void * 參數可以說是很致命的失敗設計。 另外,如果你的 B::B_API() 本身沒有存取到任何成員變數的話, 你直接把 B::API() 標成 static 讓它成為靜態成員函式就行了, 靜態成員函式指標和一般的函式指標是一樣的東西,你可以直接傳 &B::B_API 出去。 如果 B::B_API() 會存取非靜態資料成員或呼叫非靜態成員函式, 那麼你就還是乖乖去照上一段的方法去做。 這篇給你的建議很基礎,沒有用到什麼現代化的技巧,但也能幫助你瞭解問題本質。 -- Ling-hua Tseng Architect Research & Development Department Sky Mirror Technology Corporation -- ※ 發信站: 批踢踢實業坊(ptt.cc), 來自: 220.135.18.57 (臺灣) ※ 文章網址: https://www.ptt.cc/bbs/C_and_CPP/M.1601558652.A.D13.html
protoss: 10/01 23:07
legendmtg: 現在一年都不見得能看到一篇大大的文!!! <(_ _)> 10/02 10:02
Sirctal: 好久沒看到大大的文章了 10/02 17:52
sighAll: 謝謝! 10/21 22:51