全局變量可不可以定義在可被多個(gè)C文件包含的頭文件中?
用C語言編寫程序的時(shí)候,我們經(jīng)常會(huì)遇到這樣一種情況:希望在頭文件中定義一個(gè)全局變量,然后包含到兩個(gè)不同的c文件中,希望這個(gè)全局變量能在兩個(gè)文件中共用。舉例說明:項(xiàng)目文件夾project下有main.c、common.c和common.h三個(gè)文件,其中common.h文件分別#include在main.c和common.c文件中。現(xiàn)在希望聲明一個(gè)字符型變量key,在main.c和common.c中公用。 有人想,既然是想兩個(gè)文件都用,那就在common.h中聲明一個(gè)unsigned char key,然后由于包含關(guān)系,在main.c和common.c中都是可見的,所以就能共用了。 想起來確實(shí)有道理,但是實(shí)際寫出來,我們發(fā)現(xiàn)編譯的時(shí)候編譯器提示出錯(cuò),一般提示大概都類似于:Error: L6200E: Symbol key multiply defined (by common.o and main.o).也就是說編譯器認(rèn)為我們重復(fù)定義了key這個(gè)變量。這是因?yàn)?include命令就是原封不同的把頭文件中的內(nèi)容搬到#include的位置,所以相當(dāng)于main.c和common.c中都執(zhí)行了一次unsigned char key,而C語言中全局變量是項(xiàng)目內(nèi)(或者叫工程內(nèi))可見的,這樣就造成了一個(gè)項(xiàng)目中兩個(gè)變量key,編譯器就認(rèn)為是重復(fù)定義。 正確的解決辦法:使用extern關(guān)鍵字來聲明變量為外部變量。具體說就是在其中一個(gè)c文件中定義一個(gè)全局變量key,然后在另一個(gè)要使用key這個(gè)變量的c文件中使用extern關(guān)鍵字聲明一次,說明這個(gè)變量為外部變量,是在其他的c文件中定義的全局變量。請(qǐng)注意我這里的用詞:定義和聲明。例如在main.c文件中定義變量key,在common.c文件中聲明key變量為外部變量,這樣這兩個(gè)文件中就能共享這個(gè)變量key了。 (1)main.c文件#include "common.h"unsigned char key;(2)common.c文件:#include "common.h"extern unsigned char key;其實(shí)就是變量定義和變量聲明的區(qū)別,變量定義使用“數(shù)據(jù)類型+變量名稱”的形式,編譯器需要給他分配內(nèi)存單元的;而變量聲明使用“extern 變量類型+變量名稱”的形式,是告訴編譯器我這個(gè)變量將在其他外部c文件中定義,我這里只是在外部用它。編譯器就不給他分配內(nèi)存空間,而等到真正遇到變量定義的時(shí)候再給他分配內(nèi)存空間。1、普通變量定義成全局變量如果是普通類型,完全可以不用*.h文件,直接在*.c文件中定義,在調(diào)用文件處用extern 聲明,因?yàn)閷?duì)于普通類型,編譯器是可以識(shí)別的。比如在一個(gè) my.c文件中,我定義了char name[10];那么在別的文件中只要用extern char name[](由于是聲明,一位數(shù)組可以省略大小,但不建議用指針,比較指針和數(shù)組是兩回事)外部聲明就可以了,告訴編譯器這個(gè)變量我已經(jīng)定義過了,具體怎樣,你慢慢找吧。這符合常理,因?yàn)閏har是編譯器能自主識(shí)別的類型。2、自定義結(jié)構(gòu)體類型定義成全局變量不同于普通類型,如果不預(yù)先通知編譯器,編譯器是不會(huì)識(shí)別你自定義的類型的。這個(gè)時(shí)候,*.h文件便出現(xiàn)了。不是定義結(jié)構(gòu)類型不占內(nèi)存嗎?那好,我大結(jié)構(gòu)體的定義放在*.h文件中,這樣一來,無論你incude無數(shù)次,內(nèi)存都不會(huì)被占用的。而且這樣還有個(gè)好處,在別的文件中可以include這個(gè)*.h文件,這樣,在這個(gè)文件中,編譯器就可以識(shí)別你的自定義類型了,目的不就達(dá)到了? 假如我在global.h中定義了typedef struct _POSITION{int x;int y;}POSITION;那么我可以在一個(gè)global.c文件中實(shí)現(xiàn)全局變量的定義,不過要include那個(gè)*.h文件,比如/* ***global.c ******* */ include “global.h”POSITION current,;這樣就定義了cunrrent這個(gè)變量,在別的文件中引用這個(gè)變量時(shí),只要extern POSITION current;進(jìn)行聲明,然后就可以用了,不過這個(gè)文件也還得include "global.h" 因?yàn)槿绻话谶@個(gè)文件中是不識(shí)別POSITION類型的。1.如何引用一個(gè)已經(jīng)定義過的全局變量? 答:extern 可以用引用頭文件的方式,也可以用extern關(guān)鍵字,如果用引用頭文件方式來引用某個(gè)在頭文件中聲明的全局變理,假定你將那個(gè)編寫錯(cuò)了,那么在編譯期間會(huì)報(bào)錯(cuò),如果你用extern方式引用時(shí),假定你犯了同樣的錯(cuò)誤,那么在編譯期間不會(huì)報(bào)錯(cuò),而在連接期間報(bào)錯(cuò)。2.全局變量可不可以定義在可被多個(gè).C文件中?為什么? 答:可以,在不同的C文件中以static形式來聲明同名全局變量。 可以在不同的C文件中聲明同名的全局變量,前提是其中只能有一個(gè)C文件中對(duì)此變量賦初值,此時(shí)連接不會(huì)出錯(cuò)。=======================================“我是一名從事了10年開發(fā)的老程序員,最近我花了一些時(shí)間整理關(guān)于C語言、C++,自己有做的材料的整合,一個(gè)完整的學(xué)習(xí)C語言、C++的路線,學(xué)習(xí)材料和工具。全球最大的C/C++、編程愛好者的聚集地就在我這里 <進(jìn)入下方專欄即可看到及領(lǐng)取>!歡迎初學(xué)和進(jìn)階中的小伙伴。希望你也能憑自己的努力,成為下一個(gè)優(yōu)秀的程序員。工作需要、感興趣、為了入行、轉(zhuǎn)行需要學(xué)習(xí)C/C++的伙伴可以跟我一起學(xué)習(xí)!”關(guān)注我的專欄,帶你遨游代碼世界!程序猿最后分享一張C/C++學(xué)習(xí)路線圖給愛學(xué)習(xí)的小伙伴們