c與c++之間的相互調(diào)用及函數(shù)區(qū)別示例詳解
引言
最近項目需要使用google test(以下簡稱為gtest)作為單元測試框架,但是項目本身過于龐大,main函數(shù)無從找起,需要將gtest框架編譯成靜態(tài)庫使用。因為項目本身是通過純c語言編寫,而gtest則是一個c++編寫的測試框架,其中必然涉及c與c++之間的相互調(diào)用。
注意,本文的前提是,c代碼采用gcc等c語言編譯器編譯c代碼,采用g++等c++編譯器編譯c++代碼,如果c和c++代碼統(tǒng)一使用g++編譯,大部分情況是可以實現(xiàn)兩者代碼相互調(diào)用的。以下為踩坑過程的總結(jié)o_O||。
c與c++的函數(shù)區(qū)別
要了解兩者之間如何實現(xiàn)相互調(diào)用,必須先了解c與c++之間的函數(shù)有什么不同。
c++作為c語言的升級版,兩者必然有很多不同之處。其中有一個重大不同點就是,c++支持函數(shù)重載,而c語言不支持。為了使函數(shù)支持重載,c++在c語言的基礎(chǔ)上,將函數(shù)名添加上返回值和參數(shù)的類型信息。
例如,int add(int, int)
這個函數(shù),通過c++編譯器編譯后,可能呈現(xiàn)的函數(shù)名為int int_add_int_int(int, int)
(注:此處為大概地說明c++是如何將返回值和參數(shù)信息添加到函數(shù)名中的,實際中編譯器不一定是這樣實現(xiàn)的)。
從以上說明可以得出,由于c++對函數(shù)重載的支持,使得編譯后的函數(shù)符號與c語言的不一致,即使是在兩者函數(shù)名相同的前提下。
extern "C"的作用
那么,c與c++是不能相互調(diào)用了嗎?答案是否定的,因為存在著extern "C"
這個關(guān)鍵字可以使語句可以按照類C的編譯和連接規(guī)約來編譯和連接,而不是C++的編譯的連接規(guī)約。這樣在類C的代碼中就可以調(diào)用C++的函數(shù)or變量等。
注意:extern "C"
指令中的"C",表示的一種編譯和連接規(guī)約,而不是一種語言。"C"表示符合C語言的編譯和連接規(guī)約的任何語言,如Fortran、assembler等。
還有要說明的是,extern "C"
指令僅指定編譯和連接規(guī)約,但不影響語義。例如在函數(shù)聲明中,指定了extern "C"
,仍然要遵守C++的類型檢測、參數(shù)轉(zhuǎn)換規(guī)則。
c++中調(diào)用c代碼
對于c++,由于c++的編譯器對c語言兼容,因此在c++中調(diào)用c語言編寫的函數(shù),只需要在函數(shù)聲明前面加上關(guān)鍵字extern "C"
,表示采用類c語言的方式解析函數(shù)符號。例子如下:
// add.h #ifdef __ADD_H__ #define __ADD_H__ extern "C" int add(int a, int b); #endif // add.c int add(int a, int b) { return a + b; } // main.cc #include <iostream> #include "add.h" using namespace std; int main() { cout << "1 + 1 = " << add(1, 1) << endl; }
在例子中,main.cc為c++代碼,add.c為c語言代碼,當(dāng)c++編譯器識別到extern "C"
`關(guān)鍵字時,會去尋找add函數(shù)的實現(xiàn)而不是尋找類似int_add_int_int這樣帶參數(shù)信息的函數(shù)實現(xiàn)。
c語言調(diào)用c++代碼
c語言調(diào)用c++代碼卻并不容易,原因是c語言并不兼容c++。就算c語言可以調(diào)用c++,也會因為無法識別c++新定義的符號而編譯報錯。因此,為了實現(xiàn)c語言調(diào)用c++函數(shù),必須實現(xiàn)以下兩個步驟:1. 將c++相關(guān)函數(shù)封裝為靜態(tài)庫或動態(tài)庫(因為調(diào)用庫函數(shù)時編譯器并不知道里面執(zhí)行的是什么語言);2. 對外提供遵循類c語言規(guī)約的接口函數(shù)。例子如下所示:
// printNum.h #ifdef __PRINTNUM_H__ #define __PRINTNUM_H__ extern "C" void printNum(int a); #endif // printNum.cc #include <iostream> #include "printNum.h" using namespace std; void printNum(int a) { cout << << "num is " << a << endl; } // main.c extern void printNum(int a); printNum(5);
通過將cout函數(shù)封裝為類c語言規(guī)約的接口函數(shù),使得main.c中可以成功調(diào)用c++函數(shù)printNum。值得注意的是,main.c不可以直接引入printNum.h,因為c語言不能識別extern "C"
關(guān)鍵字??梢岳胏++預(yù)定義宏實現(xiàn)頭文件的改寫:
#ifdef __PRINTNUM_H__ #define __PRINTNUM_H__ #ifdef __cplusplus extern "C" { #endif void printNum(int a); #ifdef __cplusplus } #endif #endif
小結(jié)
小結(jié)如下:
- c語言與c++的相互調(diào)用可以通過
extern "C"
關(guān)鍵字實現(xiàn) - c++中調(diào)用c代碼,只須在c++中為c代碼函數(shù)聲明之前加上
extern "C"
- c語言調(diào)用c++代碼,則需要將c++代碼編譯成靜態(tài)庫或動態(tài)庫,然后對外提供用
extern "C"
聲明的類c封裝函數(shù)
以上就是c與c++之間的相互調(diào)用及函數(shù)區(qū)別示例詳解的詳細內(nèi)容,更多關(guān)于c c++相互調(diào)用的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Cocos2d-x UI開發(fā)之CCControlPotentiometer控件類使用實例
這篇文章主要介紹了Cocos2d-x UI開發(fā)之CCControlPotentiometer控件類使用實例,本文代碼中包含注釋來講解CCControlPotentiometer控件類的使用,需要的朋友可以參考下2014-09-09C++實現(xiàn)將s16le的音頻流轉(zhuǎn)換為float類型
這篇文章主要為大家詳細介紹了如何利用C++實現(xiàn)將s16le的音頻流轉(zhuǎn)換為float類型,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起了解一下2023-04-04