1.extern “C”
这个语法是c++的语法。我们知道在一个.c文件中调用另一个.c中实现的函数是没有任何问题的,一个.cpp文件调用另一个.cpp文件中实现的函数也是没有问题的。但是我们如果想要在一个.cpp文件调用另一个.c文件中实现的函数,或者在一个.c文件中调用一个.cpp实现的函数,就会出现问题,下面我们先看看以下两种错误案例。
-
cpp调用c的错误案例
Test.cpp
#include <iostream> int Add(int x, int y); int main() { int ret = Add(1, 2); std::cout << ret << std::endl; return 0; }
Add.c
int Add(int x, int y) { return x + y; }
这段代码在编译时会报错,如下
-
c调用cpp的错误案例
Test.c
#include <stdio.h> int Add(int x, int y); int main() { int ret = Add(1, 2); printf("%d\n", ret); return 0; }
Add.cpp
int Add(int x, int y) { return x + y; }
这段代码同样也会报错,如下
我们发现,这两次报错都是链接错误,这是为什么呢?
- 首先我们知道c++支持函数重载,而c语言不支持函数重载,这是由于c++的函数名修饰规则和参数类型有关,在链接的时候,能够找到对应的函数,而c语言是通过函数名去找,假如存在同名函数,编译器就不知道链接哪个函数了,所以c语言不支持函数重载
- .cpp文件在链接.c文件中的函数时,通过修饰后的函数名,发现找不到,自然会出现链接错误,而.c文件在链接.cpp文件中的函数时,根据函数名去找,自然也找不到,所以也会出现链接错误。
2.extern "C"的作用
extern "C"就可以解决上面的问题,在函数前面加extern "C"意思是告诉编译器,链接的时候按照C的函数名修饰规则去找
-
解决.cpp调用.c
Test.cpp
#include <stdio.h> extern "C" int Add(int x, int y); int main() { int ret = Add(1, 2); printf("%d\n", ret); return 0; }
Add.c
int Add(int x, int y) { return x + y; }
在链接的时候,由于使用了extern “C”,编译器就知道按照原函数名去找了,就能找到了
-
解决.c调用.cpp
Test.c
#include <stdio.h> int Add(int x, int y); int main() { int ret = Add(1, 2); printf("%d\n", ret); return 0; }
Add.cpp
extern "C" int Add(int x, int y) { return x + y; }
或者我们可以使用条件编译这样写
#ifdef __cplusplus #define EXTERN_C extern "C" #else #define EXTERN_C #endif EXTERN_C int Add(int x, int y) { return x + y; }
总之,我们想解决这种链接出错的问题,只能在.cpp文件上使用extern “C”,因为这是c++的语法,c语言可没有这种语法