在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來產生匯編碼,匯編碼可能不一樣,但意思是一樣的。