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