概述
在很多情況下,尤其是讀別人所寫**的時候,對c語言宣告的理解能力變得非常重要,而c語言本身的凝練簡約也使得c語言的宣告常常會令人感到非常困惑,因此,在這裡我用一篇的內容來集中闡述一下這個問題。
問題:宣告與函式
有一段程式儲存在起始位址為0的一段記憶體上,如果我們想要呼叫這段程式,請問該如何去做?
答案答案是(*(void (*)( ) )0)( )。看起來確實令人頭大,那好,讓我們知難而上,從兩個不同的途徑來詳細分析這個問題。
答案分析:從尾到頭
首先,最基本的函式宣告:void function (paramlist);
最基本的函式呼叫:function(paramlist);
鑑於問題中的函式沒有引數,函式呼叫可簡化為 function();
其次,根據問題描述,可以知道0是這個函式的入口位址,也就是說,0是乙個函式的指標。使用函式指標的函式宣告形式是:void (*pfunction)(),相應的呼叫形式是: (*pfunction)(),則問題中的函式呼叫可以寫作:(*0)( )。
第三,大家知道,函式指標變數不能是乙個常數,因此上式中的0必須要被轉化為函式指標。
我們先來研究一下,對於使用函式指標的函式:比如void (*pfunction)( ),函式指標變數的原型是什麼? 這個問題很簡單,pfunction函式指標原型是( void (*)( ) ),即去掉變數名,清晰起見,整個加上()號。
所以將0強制轉換為乙個返回值為void,引數為空的函式指標如下:( void (*)( ) )。
ok,結合2)和3)的分析,結果出來了,那就是:(*(void (*)( ) )0)( ) 。
答案分析:從頭到尾理解答案
(void (*)( )) ,是乙個返回值為void,引數為空的函式指標原型。
(void (*)( ))0,把0轉變成乙個返回值為void,引數為空的函式指標,指標指向的位址為0.
*(void (*)( ))0,前面加上*表示整個是乙個返回值為void的函式的名字
(*(void (*)( ))0)( ),這當然就是乙個函式了。
我們可以使用typedef清晰宣告如下:
typedef void (*pfun)( );
這樣函式變為 (*(pfun)0 )( );
問題:三個宣告的分析
對宣告進行分析,最根本的方法還是模擬替換法,從那些最基本的宣告上進行模擬,簡化,從而進行理解,下面通過分析三個例子,來具體闡述如何使用這種方法。
#1:int* (*a[5])(int, char*);
首先看到識別符號名a,「」優先順序大於「*」,a與「[5]」先結合。所以a是乙個陣列,這個陣列有5個元素,每乙個元素都是乙個指標,指標指向「(int, char*)」,很明顯,指向的是乙個函式,這個函式引數是「int, char*」,返回值是「int*」。ok,結束了乙個。:)
#2:void (*b[10]) (void (*)());
b是乙個陣列,這個陣列有10個元素,每乙個元素都是乙個指標,指標指向乙個函式,函式引數是「void (*)()」【注10】,返回值是「void」。完畢!
注意:這個引數又是乙個指標,指向乙個函式,函式引數為空,返回值是「void」。
#3. doube(*)() (*pa)[9];
pa是乙個指標,指標指向乙個陣列,這個陣列有9個元素,每乙個元素都是「doube(*)()」(也即乙個函式指標,指向乙個函式,這個函式的引數為空,返回值是「double」)。
C語言宣告
怎樣定義和宣告全域性變數和函式最好。首先,儘管乙個全域性變數或函式可以有多處宣告但是定義卻最多隻允許出現一次,對於全域性變數,定義是真正分配空間並賦初值的宣告,對於函式,定義是提供函式體的宣告。例如 這些是宣告 extern int i extern int f 而這些是定義 int i 1 int...
C語言 宣告
1 首先著眼於識別符號 變數名或者函式名 2 從距離識別符號最近的地方開始,依照優先順序解釋派生型別 指標,陣列和函式 優先順序如下 1 用於整理宣告內容的括弧 2 用於表示陣列的,用於表示函式的 3 用於表示指標的 例如int func p double func p是指向返回int型別的函式的指...
C語言宣告解析
首先,來看乙個簡單的例子 int a1 a1是乙個int int a2 a2是乙個陣列,它的每乙個元素是乙個int int a3 a3是乙個陣列,它的每乙個元素是乙個int 即,它的每乙個元素是乙個指向int的指標 int a4 a4是乙個指標,它指向乙個int陣列 可能上面的3 4兩行比較容易混淆...