在c++中,extern關鍵字用來聲明變數和函數,在聲明函數的時候,有和沒有extern的效果一樣,即下面兩條語句具有同樣的效果:
複製代碼 代碼如下:extern void fun();
void fun();
但是對於變數,有和沒有extern就有區別,當有extern時,只是告知編譯器存在這個變數,編譯器並不為該變數分配儲存空間,即真正的聲明;若沒有extern,則在聲明的同時,編譯器也為該變數分配儲存空間。
下面是有extern的情形時的c++源碼:
複製代碼 代碼如下:int main() {
extern int i;
}
下面是對應的彙編碼:
複製代碼 代碼如下:; 1 : int main() {
push ebp
mov ebp, esp;esp為指向棧頂的一個寄存器,始終指向棧頂 ebp也是一個寄存器,用來在給main函數分配的棧空間上尋訪局部變數,因此常常作為基址
;上面兩句的作用是將前一個棧的基址儲存(壓棧),然後讓ebp指向當前函數的棧空間,再次作為基址
; 2 : extern int i;
; 3 : }
xor eax, eax
pop ebp
ret 0;這三句是用來退棧用,以及函數的返回
從上面的彙編碼可以看出,並沒有為變數i在站上分配任何儲存空間
下面是沒有extern的情形的c++源碼:
複製代碼 代碼如下:int main() {
int i;
}
下面是對應的彙編碼:
複製代碼 代碼如下:; 1 : int main() {
push ebp
mov ebp, esp
push ecx;與有extern時的最大的不同就是這一句
;ecx也是一個寄存器,這裡講ecx的值壓棧,等同於為變數i在棧上分配了儲存空間
;由於ecx中的值不確定,因此,如果我們訪問沒有初始化的局部變數,常常得到一個奇怪的值
; 2 : int i;
; 3 : }
xor eax, eax
mov esp, ebp
pop ebp
ret 0
可以看出,沒有extern關鍵字時,確實為變數i在棧上分配了儲存空間
上面的彙編使用cl指令在命令列產生的,如果用vs2010來產生彙編碼,彙編碼可能不一樣,但意思是一樣的。