Posted on 2008-04-10 10:11
orlando 閱讀(177)
評論(0) 編輯 收藏 引用
1 基本解釋
extern可以置于變量或者函數(shù)前,以標(biāo)示變量或者函數(shù)的定義在別的文件中,提示編譯器遇到此變量和函數(shù)時在其他模塊中尋找其定義。
另外,extern也可用來進行鏈接指定。
2 問題:extern 變量
在一個源文件里定義了一個數(shù)組:char a[6];
在另外一個文件里用下列語句進行了聲明:extern char *a;
請問,這樣可以嗎?
答案與分析:
1)、不可以,程序運行時會告訴你非法訪問。原因在于,指向類型T的指針并不等價于類型T的數(shù)組。extern char *a聲明的是一個指針變量而不是字符數(shù)組,因此與實際的定義不同,從而造成運行時非法訪問。應(yīng)該將聲明改為extern char a[ ]。
2)、例子分析如下,如果a[] = "abcd",則外部變量a=0x61626364 (abcd的ASCII碼值),*a顯然沒有意義
顯然a指向的空間(0x61626364)沒有意義,易出現(xiàn)非法內(nèi)存訪問。
3)、這提示我們,在使用extern時候要嚴(yán)格對應(yīng)聲明時的格式,在實際編程中,這樣的錯誤屢見不鮮。
4)、extern用在變量聲明中常常有這樣一個作用,你在*.c文件中聲明了一個全局的變量,這個全局的變量如果要被引用,就放在*.h中并用extern來聲明。
3 問題:extern 函數(shù)1
常常見extern放在函數(shù)的前面成為函數(shù)聲明的一部分,那么,C語言的關(guān)鍵字extern在函數(shù)的聲明中起什么作用?
答案與分析:
如果函數(shù)的聲明中帶有關(guān)鍵字extern,僅僅是暗示這個函數(shù)可能在別的源文件里定義,沒有其它作用。即下述兩個函數(shù)聲明沒有明顯的區(qū)別:
extern int f(); 和int f();
當(dāng)然,這樣的用處還是有的,就是在程序中取代include “*.h”來聲明函數(shù),在一些復(fù)雜的項目中,我比較習(xí)慣在所有的函數(shù)聲明前添加extern修飾。
4 問題:extern 函數(shù)2
當(dāng)函數(shù)提供方單方面修改函數(shù)原型時,如果使用方不知情繼續(xù)沿用原來的extern申明,這樣編譯時編譯器不會報錯。但是在運行過程中,因為少了或者多了輸入?yún)?shù),往往會照成系統(tǒng)錯誤,這種情況應(yīng)該如何解決?
答案與分析:
目前業(yè)界針對這種情況的處理沒有一個很完美的方案,通常的做法是提供方在自己的xxx_pub.h中提供對外部接口的聲明,然后調(diào)用方include該頭文件,從而省去extern這一步。以避免這種錯誤。
寶劍有雙鋒,對extern的應(yīng)用,不同的場合應(yīng)該選擇不同的做法。
5 問題:extern “C”
在C++環(huán)境下使用C函數(shù)的時候,常常會出現(xiàn)編譯器無法找到obj模塊中的C函數(shù)定義,從而導(dǎo)致鏈接失敗的情況,應(yīng)該如何解決這種情況呢?
答案與分析:
C++語言在編譯的時候為了解決函數(shù)的多態(tài)問題,會將函數(shù)名和參數(shù)聯(lián)合起來生成一個中間的函數(shù)名稱,而C語言則不會,因此會造成鏈接時找不到對應(yīng)函數(shù)的情況,此時C函數(shù)就需要用extern “C”進行鏈接指定,這告訴編譯器,請保持我的名稱,不要給我生成用于鏈接的中間函數(shù)名。
下面是一個標(biāo)準(zhǔn)的寫法:
//在.h文件的頭上
#ifdef __cplusplus
#if __cplusplus
extern "C"{
#endif
#endif /* __cplusplus */
…
…
//.h文件結(jié)束的地方
#ifdef __cplusplus
#if __cplusplus
}
#endif
#endif /* __cplusplus */