1.問題
main.h
#ifndef _MAIN_H #define _MAIN_H unsigned char i; #endif
main.c
#include "main.h" main() { ; }
a.c
#include "main.h" 略
然后編譯a.c和main.c,就會提示Symbol i multiply defined(by a.o and main.o)
2.問題分析
2.1 #ifndef不是已經預防重復編譯了?
#ifndef #define #endif防止的是“重復編譯”,而不是“重復定義”。
重復編譯可能造成重復定義,但重復定義的來源不只有重復編譯。
從代碼變成可執行的程序,需要兩個步驟
編譯和鏈接
編譯開始時,將所有#include頭文件的地方替換成該頭文件的代碼
在編譯階段,編譯所有源文件成為模塊,各模塊中的每個變量與函數都得到了屬於自己的空間
在鏈接階段,各個模塊被組合到一起
#ifndef能夠防止在編譯階段,一段代碼被重復編譯,並且由此可以避免一個變量被重復定義
但它不能防止鏈接階段,各模塊中都有叫某個名字的變量,於是報鏈接錯誤:變量重復定義
3.解決方法
不僅用#ifndef組合防止重復編譯,而且將變量在源文件中定義,只在頭文件里放extern聲明。
這樣各模塊在編譯的時候,就知道“有這么個變量,但它的空間不在我這里”,鏈接的時候,這個變量雖然出現在所有包含這個頭文件的模塊里,但只有一個模塊是它的真身所在。