C++中extern "C"的作用是什么?(实现C和C++混合编译)

extern "C" 解决C++与C混合编译的函数名修饰问题,防止C++编译器对C函数进行name mangling,确保符号匹配;需包裹头文件或单个函数声明,且不适用于模板、inline函数、成员函数及非C兼容类型。

extern "C" 解决的是函数名修饰问题

C++ 支持函数重载,编译器在生成目标文件时会对函数名做 name mangling(名字修饰),比如 void foo(int) 可能变成 _Z3fooi;而 C 语言不重载,函数名在符号表里就是原样 foo。如果 C++ 代码直接调用 C 编译出来的库(如 libc 或自写 .a/.so),链接器找不到匹配的符号,就会报 undefined reference to 'foo' 这类错误。

在 C++ 中声明 C 函数必须用 extern "C"

告诉 C++ 编译器:这部分声明不要做 name mangling,按 C 的规则导出/导入符号。常见写法:

extern "C" {
#include 
#include "my_c_header.h"
}

或者单个函数:

extern "C" void c_function(int

x);

注意:extern "C" 是 C++ 语法,不能写在纯 C 文件里(会编译失败);#include 必须放在 extern "C" 块内才生效。

C 头文件要兼容 C++ 时需加条件宏

如果你写的 C 库希望同时被 C 和 C++ 项目包含,头文件开头要加:

#ifdef __cplusplus
extern "C" {
#endif

// 函数声明
void my_util_func(double);

#ifdef __cplusplus
}
#endif

否则 C++ 项目 #include 该头文件时,函数仍会被 mangling,链接失败。很多系统头文件(如 string.h)都这么处理。

extern "C" 不能修饰模板或内联函数

extern "C" 要求符号名唯一且无重载语义,所以:

  • 不能用于 C++ 模板函数(实例化后有多个符号)
  • 不能用于 inline 函数(通常不生成外部符号)
  • 不能用于类成员函数(C 没有类概念)
  • 参数类型必须是 C 兼容类型(比如不能传 std::stringstd::vector

传参建议用 const char*intstruct(POD 类型)、指针等。

C 和 C++ 混合编译真正容易卡住的地方,往往不是 extern "C" 写没写,而是头文件是否被正确包裹、C 库是否真的用 C 编译器(而非 g++)生成、以及结构体对齐或 ABI 版本不一致导致的运行时崩溃。