extern “C”用于c++中按c语言方式编译函数,避免名称修饰,实现C/C++混合编程。通过#ifdef __cplusplus包裹声明,确保头文件兼容C与C++,使C++可调用C函数或被C链接。
是为了实现C++代码与C语言代码的兼容。由于C++支持函数重载、类、命名空间等特性,编译器会对函数名进行“名称修饰(name mangling)”,而C语言不会。这导致C++编译后的函数符号名与C不同,无法直接被C代码调用或链接。
通过 extern “C”,可以告诉C++编译器:这部分函数按照C语言的方式进行编译和链接,即不进行名称修饰,从而实现跨语言调用。
基本语法
extern “C” { // 声明C风格的函数,用于与C代码交互 void my_c_function(int x); int add(int a, int b); }
上述代码块中的函数声明将使用C语言的链接规则。注意:extern “C” 只能用于C++代码中,不能出现在纯C文件里。
在头文件中正确使用 extern “C”
当你希望一个头文件既能被C++包含,也能被C编译器使用时,需要做条件判断:
立即学习“C++免费学习笔记(深入)”;
ifndef MY_HEADER_H
define MY_HEADER_H
ifdef __cplusplus
extern “C” {
endif
// C函数声明
void greet(void);
int compute_sum(int a, int b);
ifdef __cplusplus
}
endif
endif // MY_HEADER_H
说明:
- __cplusplus 是C++编译器定义的宏,C编译器没有这个宏。
- 这样写可保证C++编译器用 extern “C” 包裹声明,而C编译器则忽略这部分预处理指令。
在C++中调用C函数
假设你有一个C语言实现的函数:
/* math.c */
int add(int a, int b) {
return a + b;
}
在C++文件中调用它:
extern “C” {
int add(int a, int b);
}
#include <iostream>
int main() {
std::cout return 0;
}
确保编译时将C文件和C++文件一起链接:
gcc -c math.c -o math.o
g++ main.cpp math.o -o program
在C中调用C++函数(反向调用)
C语言无法直接调用C++函数,尤其是带重载或类成员函数的。但可以通过以下方式间接实现:
// wrapper.cpp
include <iostream>
class Math {
public:
static int multiply(int a, int b) {
return a * b;
}
};
extern “C” {
int call_multiply(int a, int b) {
return Math::multiply(a, b);
}
}
然后在C代码中声明并调用:
/* main.c */
extern int call_multiply(int a, int b);
int main() {
int result = call_multiply(5, 6);
printf(“Result: %dn”, result);
return 0;
}
编译链接:
g++ -c wrapper.cpp -o wrapper.o
gcc main.c wrapper.o -o program
关键点:必须用 extern “C” 定义一个C++函数的包装函数,该包装函数本身是C兼容的。
基本上就这些。extern “C” 不改变函数行为,只影响编译和链接方式。正确使用它,就能在C和C++之间搭建桥梁。注意别在C++类成员函数或重载函数上直接使用 extern “C”,那会出错。包装一层最稳妥。


