extern用法详解
[轉]extern用法詳解
Posted on 2011-08-16 11:15 單魚游弋 閱讀(98) 評論(0)編輯收藏1 基本解釋
   extern可以置于變量或者函數前,以標示變量或者函數的定義在別的文件中,提示編譯器遇到此變量和函數時在其他模塊中尋找其定義。
   另外,extern也可用來進行鏈接指定。
2 問題:extern 變量
   在一個源文件里定義了一個數組:char a[6];
   在另外一個文件里用下列語句進行了聲明:extern char *a;
   請問,這樣可以嗎? 
   答案與分析:
   1)、不可以,程序運行時會告訴你非法訪問。原因在于,指向類型T的指針并不等價于類型T的數組。extern char *a聲明的是一個指針變量而不是字符數組,因此與實際的定義不同,從而造成運行時非法訪問。應該將聲明改為extern char a[ ]。
   2)、例子分析如下,如果a[] = "abcd",則外部變量a=0x61626364 (abcd的ASCII碼值),*a顯然沒有意義
   顯然a指向的空間(0x61626364)沒有意義,易出現非法內存訪問。
   3)、這提示我們,在使用extern時候要嚴格對應聲明時的格式,在實際編程中,這樣的錯誤屢見不鮮。
   4)、extern用在變量聲明中常常有這樣一個作用,你在*.c文件中聲明了一個全局的變量,這個全局的變量如果要被引用,就放在*.h中并用extern來聲明。
 
4 問題:extern  函數2
   當函數提供方單方面修改函數原型時,如果使用方不知情繼續沿用原來的extern申明,這樣編譯時編譯器不會報錯。但是在運行過程中,因為少了或者多了輸入參數,往往會照成系統錯誤,這種情況應該如何解決?
   答案與分析:
   目前業界針對這種情況的處理沒有一個很完美的方案,通常的做法是提供方在自己的xxx_pub.h中提供對外部接口的聲明,然后調用方include該頭文件,從而省去extern這一步。以避免這種錯誤。
   寶劍有雙鋒,對extern的應用,不同的場合應該選擇不同的做法。
5 問題:extern “C”
   在C++環境下使用C函數的時候,常常會出現編譯器無法找到obj模塊中的C函數定義,從而導致鏈接失敗的情況,應該如何解決這種情況呢?
  答案與分析:
   C++語言在編譯的時候為了解決函數的多態問題,會將函數名和參數聯合起來生成一個中間的函數名稱,而C語言則不會,因此會造成鏈接時找不到對應函數的情況,此時C函數就需要用extern “C”進行鏈接指定,這告訴編譯器,請保持我的名稱,不要給我生成用于鏈接的中間函數名。
   下面是一個標準的寫法:
 //在.h文件的頭上
 #ifdef __cplusplus
 #if __cplusplus
 extern "C"{
  #endif
  #endif /* __cplusplus */ 
  …
  …
  //.h文件結束的地方
  #ifdef __cplusplus
  #if __cplusplus
 }
 #endif
 #endif /* __cplusplus */?
3 問題:extern  函數1
   常常見extern放在函數的前面成為函數聲明的一部分,那么,C語言的關鍵字extern在函數的聲明中起什么作用?
   答案與分析:
   如果函數的聲明中帶有關鍵字extern,僅僅是暗示這個函數可能在別的源文件里定義,沒有其它作用。即下述兩個函數聲明沒有明顯的區別:
 extern int f(); 和int f();
   當然,這樣的用處還是有的,就是在程序中取代include “*.h”來聲明函數,在一些復雜的項目中,我比較習慣在所有的函數聲明前添加extern修飾。
=================================================================================================================================
用例子給你示范 // 1.cpp int x = 10; // 2.cpp 注意沒有包含1.cpp #include <iostream> using namespace std; extern int x; int main () { cout << x << endl; } //則輸出10 兩個文件同在一個項目(project)中,你不包含某個文件(cpp)而可以用它內部定義的變量,(里是.pp不是.h, 因為在.h中定義的數據不能在.cpp中用除非這個.cpp包含這個.h文件) 例: // 1.h #include <iostream> void print() { std::cout << "hello!" << std::endl; } // 2.cpp #include <iostream> using namespace std; // 以上兩句在這個例子中可以不要 extern void print(); int main () { print(); } 就會出錯因為1.h中的void print();在不包含它的文件中是不可調用的,即使在聲明了extern 也于事無補,如果你將這個例子中的1.h名字換成1.cpp就對了! 從這些可以看出來,extern在這里起的作用是告訴編譯器,你這個print()已經在某個.cpp中已經定義了,這里只不過是聲明一下有這個東西,然后拿來用一下。定義只能出現一次,聲明卻可出現多次,也就是說extern聲明可在多個文件中用(包括.h) 還有,你還可以屏蔽extern聲明,如第二個例子中的第二個.cpp文件可以改成 #include <iostream> using namespace std; // 這里以上兩句不能省略,因為,這里extern void print();函數已經不起作用了,在這里調用的而是本文件中定義的void print()函數,其中用到了cout,endl;他們來源于std::<iostream> extern void print(); void print() { cout << "world!" << endl; } int main () { print(); } // 輸出結果為world! 還有一個extern "C"就不用說了,用這個可以允許C++程序中調用C的函數!==================================================================================================================================<p style="color: rgb(54, 46, 43); font-family: Arial; font-size: 14px; line-height: 26px;">extern "C"的主要作用就是為了能夠正確實現C++代碼調用其他C語言代碼。</p><p style="color: rgb(54, 46, 43); font-family: Arial; font-size: 14px; line-height: 26px;">加上extern "C"后,會指示編譯器這部分代碼按C語言的進行編譯,而不是C++的</p><p style="color: rgb(54, 46, 43); font-family: Arial; font-size: 14px; line-height: 26px;">由于C++支持函數重載,因此編譯器編譯函數的過程中會將函數的參數類型也加到編譯后的代碼中,而不僅僅是函數名;而C語言并不支持函數重載,因此編譯C語言代碼的函數時不會帶上函數的參數類型,一般之包括函數名</p>
總結
以上是生活随笔為你收集整理的extern用法详解的全部內容,希望文章能夠幫你解決所遇到的問題。
 
                            
                        - 上一篇: 如何定位Release程序崩溃原因
- 下一篇: 关于模板函数/模板类编译成DLL
