C++調用C函數的方法
我們以前見到extern "C"這樣的語句,只是簡單地知道跟外部鏈接有關,但是沒有深刻理解它的意思。今天繼續和小編一起學習C++調用C函數的方法吧!
C++調用C函數的方法首先,為什麼要使用extern "C"修飾符?
C++調用其它語言的函數,由於編譯器生成函數的機制不一樣,所以需要經過特殊處理,才可以調用。調用C語言的函數,需要在函數聲明的地方語句extern "C"。如果不使用該語句,在鏈接的`時候,編譯器就會報以下這種錯誤。
: error LNK2019: 無法解析的外部符號 "void __cdecl DeleteStack(struct _Node *)" (?DeleteStack@@YAXPAU_Node@@@Z),該符號在函數 _main 中被引用。
然後是如何使用?
應該怎麼使用該語句呢?
剛開始,我簡單地在C++源文件的前面使用該語句聲明,但是還是出錯,而且是在編譯階段就報錯。
error C2732: 鏈接規範與“DeleteStack”的早期規範衝突。
為什麼會出現這個錯誤呢?因為C++源文件已經引入了C的頭文件,在頭文件裏,聲明該函數時沒有extern修飾,而這裏有extern修飾,所以衝突了。解決的辦法有兩個。
一。在C頭文件中加上extern修飾符。
直接加,也不行。因為C源文件也包含了這個頭文件,當編譯C源文件時,就會出現錯誤。所以,需要一種機制來區分是編譯C還是C++文件。方法如下:
#ifdef __cplusplus
extern "C"
#endif
void DeleteStack(Stack stack);
因為在編譯C++文件時,自動定義預處理器名字__cplusplus,而編譯C時,沒有該處理器名字。所以只有編譯C++時,才有符號extern “C”。
此外,鏈接指示extern "C"有單個和複合兩種形式。以上為單個形式,複合形式可以同時將幾個函數聲明為extern "C"
extern "C" {
void DeleteStack(Stack stack);
void PrintStack(Stack stack);
void Pop(Stack stack);
}
加上預處理器名字如下:
#ifdef __cplusplus
extern "C" {
#endif
void DeleteStack(Stack stack);
void PrintStack(Stack stack);
void Pop(Stack stack);
#ifdef __cplusplus
}
#endif
二。編寫一個C++風格的頭文件,在這裏添加extern修飾符。
使用方法一,很簡單。但是如果該頭文件是別人寫好,你無法修改。這個時候就要使用其它方法了。方法是定義C++自己的頭文件,文件名為"CStack.h"
// CStack.h
extern "C" {
#include "Stack.h";
}