用C語言編寫程序的時候,我們經常會遇到這樣一種情況:希望在頭文件中定義一個全局變量,然后包含到兩個不同的c文件中,希望這個全局變量能在兩個文件中共用。
舉例說明:項目文件夾project下有main.c、common.c和common.h三個文件,其中common.h文件分別#include在main.c和common.c文件中。現在希望聲明一個字符型變量key,在main.c和common.c中公用。如下圖所示:
有人想,既然是想兩個文件都用,那就在common.h中聲明一個unsigned char key,然后由于包含關系,在main.c和common.c中都是可見的,所以就能共用了。
這種想法其實是很多初學者都會想到的,想起來確實有道理,但是實際寫出來,我們發現編譯的時候編譯器提示出錯,一般提示大概都類似于:Error: L6200E: Symbol key multiply defined (by common.o and main.o). 也就是說編譯器認為我們重復定義了key這個變量。這是因為#include命令就是原封不同的把頭文件中的內容搬到#include的位置,所以相當于main.c和common.c中都執行了一次unsigned char key,而C語言中全局變量是項目內(或者叫工程內)可見的,這樣就造成了一個項目中兩個變量key,編譯器就認為是重復定義。
正確的解決辦法:使用extern關鍵字來聲明變量為外部變量。具體說就是在其中一個c文件中定義一個全局變量key,然后在另一個要使用key這個變量的c文件中使用extern關鍵字聲明一次,說明這個變量為外部變量,是在其他的c文件中定義的全局變量。請注意我這里的用詞:定義和聲明。例如在main.c文件中定義變量key,在common.c文件中聲明key變量為外部變量,這樣這兩個文件中就能共享這個變量key了,如下圖所示。
(1)main.c文件
#include "common.h"
unsigned char key;
(2)common.c文件:
#include "common.h"
extern unsigned char key;
很多人看了可能糊涂,這里稍微說一下,其實就是變量定義和變量聲明的區別,變量定義使用“數據類型+變量名稱”的形式,編譯器需要給他分配內存單元的;而變量聲明使用“extern 變量類型+變量名稱”的形式,是告訴編譯器我這個變量將在其他外部c文件中定義,我這里只是在外部用它。編譯器就不給他分配內存空間,而等到真正遇到變量定義的時候再給他分配內存空間。
由于很多人從開始學C語言就一直把定義變量叫聲明變量,一開始就叫錯了,所以導致現在分不清定義和聲明的區別。要是還理解不了就想想函數的定義和聲明,函數定義是編寫函數功能實體,編譯器要編譯這個函數并且要分配內存空間,而函數聲明并不生成函數功能實體,只是告訴編譯器這是個函數,這個函數在后面將會定義實體,我這里只是提前用,編譯器就會接著繼續往下編譯,如果子函數寫在main函數之后,那么聲明是必須的,如果不聲明函數編譯器都不知道這是個函數,編譯就會報錯。
說了這么多應該說明白了,如果還有問題可以參考譚浩強的那本C語言書,書中有詳細解釋和例子。