extern "C"的作用是让C++编译器禁用名字修饰,生成与C语言兼容的符号名以实现C/C++混合链接。它用于函数声明(如extern "C" { int f(int); })和头文件条件编译,但不可用于模板、重载函数、类成员函数、inline或static函数。

extern "C" 的作用是告诉 C++ 编译器:这部分代码要按 C 语言的方式链接,不进行 C++ 的名字修饰(name mangling)。
C++ 为什么需要 extern "C"
C++ 支持函数重载、类、命名空间等特性,编译器在生成目标文件时,会把函数名改写成带类型、作用域信息的内部符号(比如 _Z3addii),这叫名字修饰。而 C 语言没有重载,函数名在符号表里就是原样(比如 add)。如果 C++ 直接调用 C 的函数,或 C 调用 C++ 的函数,链接器找不到匹配的符号,就会报 undefined reference 错误。
加上 extern "C" 后,C++ 编译器就禁用名字修饰,生成和 C 一致的符号名,实现二进制兼容。
常见使用方式
- 在 C++ 代码中调用 C 库(如
stdio.h、stdlib.h)时,标准头文件内部通常已用extern "C"包裹,所以你不用手动加 - 自己写的 C 函数被 C++ 调用:在 C++ 文件里声明时加
extern "C",例如:
extern "C" {
int my_c_func(int x);
} - 自己写的 C 函数既要被 C 调用、也要被 C++ 调用:在 C 头文件里用宏做条件编译:
#ifdef __cplusplus
extern "C" {
#endif
int my_c_func(int x);
#ifdef __cplusplus
}
#endif
不能 extern "C" 的情况
- 不能修饰 C++ 特有语法:比如模板、重载函数、类成员函数、返回或参数含 class/struct 的非 POD 类型(除非明确是 C 兼容布局)
- 不能修饰内联函数(inline),因为 inline 展开发生在编译期,不生成外部符号
- 不能修饰静态函数(static),因为它本就不参与外部链接
混合编译的关键点
- C 源文件(.c)只能用 C 编译器(如 gcc -c),不能用 g++ 编译;否则即使没
extern "C"也可能因 ABI 差异出问题 - C++ 源文件(.cpp)可以包含 C 头文件,但该头文件必须已正确用
extern "C"保护 - 链接时,C 和 C++ 目标文件可混链(如
gcc main.o utils.o -o app),只要符号名匹配即可 - 全局变量也可以用
extern "C"声明,规则和函数一样
基本上就这些。核心就一条:让 C++ “装作”是 C,才能和 C 的世界握手。不复杂,但容易忽略。
立即学习“C语言免费学习笔记(深入)”;











