注意這里的C調用C++或者C++調用C意思是.c文件中調用.cpp文件中代碼,或者相反。
C++和C是兩種完全不同的編譯鏈接處理方式,如果直接在C++里面調用C函數,會找不到函數體,報鏈接錯誤。
要解決這個問題,就要在 C++文件里面顯示聲明一下哪些函數是C寫的,要用C的方式來處理。
1.引用頭文件前需要加上 extern “C”,如果引用多個,那么就如下所示
extern “C”
{
#include “ s.h”
#include “t.h”
#include “g.h”
#include “j.h”
};
然后在調用這些函數之前,需要將函數也全部聲明一遍。
2.C++調用C函數的方法,將用到的函數全部重新聲明一遍
extern “C”
{
extern void A_app(int);
extern void B_app(int);
extern void C_app(int);
extern void D_app(int);
}
C++程序中調用被c編譯器編譯后的函數,為什么要加extern "C"?
C++語言支持函數重載,C語言不支持函數重載。函數被C++編譯后在庫中的名字與C語言的不同。
假設某個C 函數的聲明如下:
void foo(int x, int y);
該函數被C 編譯器編譯后在庫中的名字為_foo,而C++編譯器則會產生像_foo_int_int之類的名字用來支持函數重載和類型安全連接。
由於編譯后的名字不同,C++程序不能直接調用C 函數。
C++提供了一個C 連接交換指定符號extern“C”來解決這個問題。
例如:
extern “C”
{
void foo(int x, int y); // 其它函數
}
或者寫成
extern “C”
{
#include “myheader.h” // 其它C 頭文件
}
這就告訴C++編譯譯器,函數 foo 是個C 連接,應該到庫中找名字_foo 而不是找_foo_int_int。
C++編譯器開發商已經對C 標准庫的頭文件作了extern“C”處理,所以我們可以用#include 直接引用這些頭文件。
=======================================================================
要明白為何使用extern "C",還得從cpp中對函數的重載處理開始說起。
在c++中,為了支持重載機制,在編譯生成的匯編碼中,要對函數的名字進行一些處理,加入比如函數的返回類型等等.
而在C中,只是簡單的函數名字而已,不會加入其他的信息.
也就是說:C++和C對產生的函數名字的處理是不一樣的.
明白了加入與不加入extern "C"之后對函數名稱產生的影響,我們繼續我們的討論:為什么需要使用extern "C"呢?
C++之父在設計C++之時,考慮到當時已經存在了大量的C代碼,為了支持原來的C代碼和已經寫好C庫,需要在C++中盡可能的支持C,而extern "C"就是其中的一個策略。
試想這樣的情況:一個庫文件已經用C寫好了而且運行得很良好,這個時候我們需要使用這個庫文件,但是我們需要使用C++來寫這個新的代碼。如果這個代碼使用的是C++的方式鏈接這個C庫文件的話,那么就會出現鏈接錯誤.
現在我們有了一個C庫文件,它的頭文件是f.h,產生的lib文件是f.lib,那么我們如果要在C++中使用這個庫文件,我們需要這樣寫:
extern "C" {
#include "f.h"
}
參考:http://blog.csdn.net/juanjuan888/article/details/6889843
http://www.cnblogs.com/Yogurshine/p/3913073.html
