typedef函式指標的宣告與其他型別的宣告有所不同,主要遵守右左法則,具體的解釋如下:
1.1 解讀的原則---右左原則:
右左法則:首先從最裡面的圓括號看起,然後往右看,再往左看。每當遇到圓括號時,就應該掉轉閱讀方向。一旦解析完圓括號裡面所有的東西,就跳出圓括號。重複這個過程直到整個宣告解析完畢。
個人認為最開始應該是從需要定義的識別符號開始閱讀(當然有可能所有的需要定義的識別符號都是在圓括號內的),因為乙個宣告中可能有多個識別符號,但是未定義的識別符號永遠只有乙個。
注:函式的返回值不能是乙個陣列。陣列的元素型別必須一樣,同時他們占用的空間也必須一樣,所以陣列內不能存放函式。
通過以下的例子討論右左法則:
int (*func)(int *p) ;
首先找到未定義的識別符號,就是func,它的外面有一對圓括號,而且左邊是乙個*號,說明func時乙個指標,然後跳出這個圓括號,先看右邊,也是乙個圓括號,這說明(*func)時乙個函式,而func是乙個指向這類函式的指標,就是乙個函式指標,這類函式具有int*型別 的形參,返回值型別是int。
int (*func)(int *p, int (*f)(int*));
func被一對括號包含,且左邊有乙個*號,說明func是乙個指標,跳出括號,右邊有乙個括號,那麼func時乙個指向函式的指標,這類函式具有int *和int (*)(int *)這樣的形參,返回值為int型別,看一下func的形參int (*f)(int*),類似前面的解釋,f也是乙個函式指標,指向的函式具有int*型別的形參,返回值為int。
int (*func[5])(int *p);
func右邊是乙個[ ]運算子,說明func是乙個具有5個元素的陣列,func左邊有乙個*,說明func的元素是指標,這裡的*是修飾func[5]而不是func的,應為運算子的優先順序比*高,所以func先與先結合。跳出這個括號,右邊也是一對圓括號,說明func陣列的元素是函式型別的指標,它所指向的函式具有int*型別的形參,返回值型別為int。
int (*(*func)[5])(int *p);
func被乙個圓括號包含,左邊又有乙個*,那麼func是乙個指標,跳出括號,右邊是乙個運算符號,說明func是乙個指向陣列的指標,現在往左看,左邊有乙個*號,說明這個陣列的元素是指標,再跳出括號,右邊又有乙個括號,說明這個陣列的元素是指向函式的指標。總結一下,就是:func是乙個指向陣列的指標,這個陣列的元素是函式指標,這些指標指向具有int*形參,返回值為int型別的函式。
int (*(*func)(int *p))[5];
func是乙個函式指標,這類函式具有int*型別的形參,返回值是指向陣列的指標,所指向的陣列的元素是具有5個int元素的陣列。
要注意有些複雜指標宣告是非法的,例如:
int func(void) [5];
func是乙個返回值為具有5個int元素的陣列的函式。但c語言的函式返回值不能為陣列,這是因為如果允許函式返回值為陣列,那麼接收這個陣列的內容的東西,也必須是乙個陣列,但c語言的陣列名是乙個右值,它不能作為左值來接收另乙個陣列,因此函式返回值不能為陣列。
int func[5](void);
func是乙個具有5個元素的陣列,這個陣列的元素都是函式。這也是非法的,因為陣列的元素除了型別必須一樣外,每個元素所占用的記憶體空間也必須相同,顯然函式是無法達到這個要求的,即使函式的型別一樣,但函式所占用的空間通常是不相同的。
參考:
函式指標與函式宣告
問題 當計算機啟動時,硬體將呼叫首位址為0的子例程,為了模擬開機啟動時的情形,如何設計出乙個c語句,以顯示呼叫該子例程。換個問法 有一段程式儲存在起始位址為 0的一段記憶體上,如果我們想要呼叫這段程式,請問該如何去做?c陷阱與缺陷 給出答案 void 0 相關知識 函式指標 函式指標是指向函式的指標...
詳解C C 函式指標宣告
要理解乙個c程式,僅僅理解組成該程式的符號是不夠的。程式設計師還必須理解這些符號是如何組合成宣告 表示式 語句和程式的。我們先來看看下面的乙個語句 1 void 0 這是當計算機啟動時,硬體將呼叫首位址為0位置的子例程。像這樣的表示式恐怕會令每個c c 程式設計師的內心都 不寒而慄 吧。然而,完全不...
詳解C C 函式指標宣告
要理解乙個c程式,僅僅理解組成該程式的符號是不夠的。程式設計師還必須理解這些符號是如何組合成宣告 表示式 語句和程式的。我們先來看看下面的乙個語句 1 void 0 這是當計算機啟動時,硬體將呼叫首位址為0位置的子例程。像這樣的表示式恐怕會令每個c c 程式設計師的內心都 不寒而慄 吧。然而,完全不...