關(guān)鍵字extern
我們知道C語言中extern是一個申明的關(guān)鍵字,那么申明與定義之間有什么關(guān)系呢? 其實extern int a;就是一個對變量的申明,它與定義的本質(zhì)是定義分配了真正的內(nèi)存空間(注意這里內(nèi)存是物理內(nèi)存,我們后面可能會了解到虛擬存儲中的邏輯內(nèi)存),而申明只是說明了這里的變量在別的地方定義過了,我們這里需要使用它。申明在有的地方叫做引用性的申明,定義又叫做定義性的申明或者直接的叫做定義。
1.變量的申明與定義
extern int a; 1
int a;2 ?
大家看一個就是變量的申明,而第二個就是定義。下面我們來看一下具體文件中的使用。 假如我們定義了
文件1.hello.c ?
#include
hello.h?
1.#ifndef ?_HELLO_H_
2.#define ?_HELLO_H_
3.extern void hello();
4.int a=1;
5.#endif //hello.h
main.c
1. #include
從運行的結(jié)構(gòu)我們可以看到我們可以成功的在main.c中輸出a的值,并且調(diào)用函數(shù)hello(),關(guān)于hello()是extern的函數(shù)申明,我們接下來將。我們先來分析一下上面的程序。
在hello.h 我們定義了a變量并且進行了初始化,而在main.c 中我們只需要第四句就可以操作變量,因為這里的作用就是申明這個變量我們在別的文件中已經(jīng)定義,我們才可以使用此變量。 注意:一般一個在一個文件的變量最大的作用域只是在本文件中(也就是在文件開頭定義的全局變量)。 而且這里我們可以看到我們打印出來a的值是我們在hello.h中初始化a的值,如果我們這里認為main.c中的a值是定義那么它的值輸出應該是編譯器默認輸出的0,這里顯然不是,從以上的兩點我們證明了這里是main.c的a是對變量申明,并且沒有分配內(nèi)存,只有定義才會分配內(nèi)存。
2.函數(shù)的申明
關(guān)于使用extern對函數(shù)的申明其實很簡單了,我們知道C語言是面向過程的一種語言,代碼的實現(xiàn)過程是一步一步的執(zhí)行,他不像面向?qū)ο蟮恼Z言那樣靈活使用,它是比較死板的一種語言,只會按部就班的按照過程一步一步的來執(zhí)行。
所以一個函數(shù)我們在定義并且執(zhí)行函數(shù)體之前必須對它進行一個,告訴編譯器這里有一個函數(shù)的申明,我們下面的函數(shù)的定義在這里已經(jīng)進行申明了,這樣才能保證程序的運行。因此extern修飾的函數(shù)就是對函數(shù)進行一個申明,為了我們后面使用該函數(shù)的調(diào)用準備條件,這里extern void hello(); 和void hello(); 其實都是函數(shù)的申明,只是加上關(guān)鍵字extern之后我們的代碼可讀性會更好,不加上也是可以通過的。 但是我們不寫這條語句后的分號是絕對不行的,那樣就變成了函數(shù)的定義的,這個時候編譯器老兄會讓你通過嗎? 還有不寫這條語句也不行的。呵呵,肯定不會的。不信你試試!