※ 引述《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() 會存取非靜態資料成員或呼叫非靜態成員函式,
那麼你就還是乖乖去照上一段的方法去做。
這篇給你的建議很基礎,沒有用到什麼現代化的技巧,但也能幫助你瞭解問題本質。