extern “C” 用于防止 c++ 名称修饰,使 C ++ 能调用 C 函数或导出 C 接口,解决链接时符号不匹配问题。
的主要作用是解决 C ++ 对函数名进行 ** 名称修饰(name mangling)** 的问题,从而实现 C ++ 代码调用 c 语言 函数,或者让 C 代码能够使用 C ++ 中实现的函数(前提是函数用 C 的方式链接)。这个机制是 C 和 C ++ 混合编程的基础。
为什么 需要 extern “C”?
C++ 支持 函数重载 、类、 命名空间 等特性,因此编译器在编译时会对函数名进行修饰,加入参数类型、命名空间等信息,形成唯一的符号名。而 C 语言不支持这些特性,函数名在编译后基本保持原样。
例如,C++ 中两个函数:
void func(int);
void func(double);
会被编译成不同的符号名,如 _Z4funci 和 _Z4funcd。而 C 语言中只有一个 func 符号。
立即学习“ C 语言免费学习笔记(深入)”;
如果在 C ++ 代码中直接调用一个 C 语言写的函数 void c_func();,而没有告诉编译器这是 C 风格的链接方式,链接器会去找一个经过 C ++ 修饰的符号,结果找不到,就会报 undefined reference 错误。
extern “C” 的使用方法
通过 extern “C” 可以告诉 C ++ 编译器:这部分函数应按照 C 语言的链接规则处理,即不进行名称修饰。
常见用法有两种:
- 单个函数声明:
extern “C” void c_function(); - 多个函数批量声明:
extern “C” {
void func1();
void func2();
int add(int, int);
}
通常在 C 语言头文件中,为了兼容 C ++ 编译器,会这样写:
#ifdef __cplusplus
extern “C” {
#endif
// C 函数声明
void c_api_init();
int c_api_process(int data);
#ifdef __cplusplus
}
#endif
这样,当这个头文件被 C ++ 文件包含时,编译器会用 C 的链接方式处理这些函数;被 C 编译器包含时,则忽略 extern “C” 部分(因为 C 不认识),正常编译。
实际应用场景
这种机制广泛用于系统级编程、库开发、嵌入式开发等领域。
- 调用 C 标准库 函数(如 printf、malloc)时,虽然我们没显式写 extern “C”,但系统头文件内部已经做了处理。
- 使用 C 语言编写的第三方库(如 Openssl、libcurl)时,必须确保以 C 方式链接。
- 在 C ++ 项目中 封装 C 模块,或提供 C 接口供其他语言(如python 通过 ctypes)调用。
基本上就这些。只要涉及 C 和 C ++ 混用,extern “C” 就是绕不开的关键语法,理解它有助于避免链接错误,提升跨语言协作能力。