#include <iostream>
#include <string>
using namespace std;
void alert( string msg )
{
cout << msg << endl;
}
void (*GetFunction())(string)
{
return alert;
}
int main()
{
// 剛看完C++ Primer第七章,合上書,寫個復雜點的函數試一下,剛開始函數聲明不是這樣的,一點點往上加后來才變成了void (*((*ptr)()))(string)。
//寫完后就有種預感——明天看了估計就看不懂了,趕緊記下來,呵呵。
// 正在從C#轉C++,這是在園子C++版里的第一篇文章,有錯的地方請大家不要給面子,該拍磚拍磚;-)
//ptr是一個指針,他指向一個函數,這個函數返回一個指針,該指針指向一個函數,這個函數帶有一個string參數,但是沒有返回值
void (*((*ptr)()))(string) = GetFunction;
//寫完了聲明然后調用,剛開始我寫的是(*ptr)()("Hello C++"),一運行結果出來了,挺開心,可是在重讀一遍的時候發現自己寫“錯”了,忘了解引用
可是結果怎么是對的?
//接著又寫了(*((*ptr)()))("Hi C++"),F5,結果也出來了
于是剛從函數聲明中解脫出來的大腦又暈了
(*ptr)()("Hello C++");
(*((*ptr)()))("Hi C++");
//為什么呢,難道函數指針在調用時解不解引用都可以嗎?為了證明這個結論,再聲明一個的函數指針試一下
void (*ptrAlert)(string);
ptrAlert = alert;
//不解引用
ptrAlert("Hello Alert");
//解引用
(*ptrAlert)("Hi Alert");
/*
結論:函數指針在調用時解不解引用都可以
在網上搜到的相關資料:
圍繞這兩種寫法,當初C89制定的時候曾經有過爭論。(*p)();是一種舊式的規定,舊式規定圓括號左邊必須具有“函數”類型,如果是指向函數的指針,
那么必須加上*聲明符。但C89不再把圓括號的左邊限定為“函數”類型,而是一個后綴表達式。那么問題就來了,如果p的值是函數地址,那么*號就是聲明符,
但如果p指向的內容是函數地址,*號就得被看作運算符了。同一種形式會有兩種解釋,這是一個矛盾。不僅函數調用如此,指向數組的指針也存在這種矛盾。
編譯器為了處理這種情況得增加代碼,效率自然就降低了。爭論的最后結果是誰也不能把對方完全說服,于是就干脆兩種都支持了。
說實話,我沒怎么看懂上面的話,因為不了解C89這些標準,只是覺得(*p)()容易理解些,雖然寫著麻煩點,以后c++程序寫多了應該會有更深的認識吧。
*/
}
#include <string>
using namespace std;
void alert( string msg )
{
cout << msg << endl;
}
void (*GetFunction())(string)
{
return alert;
}
int main()
{
// 剛看完C++ Primer第七章,合上書,寫個復雜點的函數試一下,剛開始函數聲明不是這樣的,一點點往上加后來才變成了void (*((*ptr)()))(string)。
//寫完后就有種預感——明天看了估計就看不懂了,趕緊記下來,呵呵。
// 正在從C#轉C++,這是在園子C++版里的第一篇文章,有錯的地方請大家不要給面子,該拍磚拍磚;-)
//ptr是一個指針,他指向一個函數,這個函數返回一個指針,該指針指向一個函數,這個函數帶有一個string參數,但是沒有返回值
void (*((*ptr)()))(string) = GetFunction;
//寫完了聲明然后調用,剛開始我寫的是(*ptr)()("Hello C++"),一運行結果出來了,挺開心,可是在重讀一遍的時候發現自己寫“錯”了,忘了解引用

//接著又寫了(*((*ptr)()))("Hi C++"),F5,結果也出來了


(*ptr)()("Hello C++");
(*((*ptr)()))("Hi C++");
//為什么呢,難道函數指針在調用時解不解引用都可以嗎?為了證明這個結論,再聲明一個的函數指針試一下
void (*ptrAlert)(string);
ptrAlert = alert;
//不解引用
ptrAlert("Hello Alert");
//解引用
(*ptrAlert)("Hi Alert");
/*
結論:函數指針在調用時解不解引用都可以
在網上搜到的相關資料:
圍繞這兩種寫法,當初C89制定的時候曾經有過爭論。(*p)();是一種舊式的規定,舊式規定圓括號左邊必須具有“函數”類型,如果是指向函數的指針,
那么必須加上*聲明符。但C89不再把圓括號的左邊限定為“函數”類型,而是一個后綴表達式。那么問題就來了,如果p的值是函數地址,那么*號就是聲明符,
但如果p指向的內容是函數地址,*號就得被看作運算符了。同一種形式會有兩種解釋,這是一個矛盾。不僅函數調用如此,指向數組的指針也存在這種矛盾。
編譯器為了處理這種情況得增加代碼,效率自然就降低了。爭論的最后結果是誰也不能把對方完全說服,于是就干脆兩種都支持了。
說實話,我沒怎么看懂上面的話,因為不了解C89這些標準,只是覺得(*p)()容易理解些,雖然寫著麻煩點,以后c++程序寫多了應該會有更深的認識吧。
*/
}
C#也不錯么!
void (*((*ptr)()))(string);
寫成void ( *(*ptr)() )(string);
也可以的,本來operator()就優先于operator*,我個人不太喜歡過多的小括號,去掉括號反而覺得簡潔,就像*it++對吧。
關于調用,感覺是不解引用的形式好,因為還有函數類,格式通用的話,使用模板的時候會很方便。
二義性也不是那么可怕,這個也是Coding下來總結出來的結果,人對代碼的感覺是神奇的,不能和區區一些編譯時間相提并論。
總覺得C#很不自由,可能是因為我的興趣從web開發轉移到圖形和瀏覽器插件上了吧。其實也不是“轉”,就是想多熟悉一個語言,我現在還得靠C#吃飯呢,呵呵。
@Alleluja
嗯,謝謝。可能以后c++代碼寫多了就有你這種感覺了,不過以我現在的程度不寫一堆括號估計一轉眼自己都看不懂了 ;-)