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++免费学习笔记(深入)”;
本文档主要讲述的是Android中JNI编程的那些事儿;JNI译为Java本地接口。它允许Java代码和其他语言编写的代码进行交互。在android中提供JNI的方式,让Java程序可以调用C语言程序。android中很多Java类都具有native接口,这些接口由本地实现,然后注册到系统中。希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以过来看看
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::string、std::vector)
传参建议用 const char*、int、struct(POD 类型)、指针等。
extern "C" 写没写,而是头文件是否被正确包裹、C 库是否真的用 C 编译器(而非 g++)生成、以及结构体对齐或 ABI 版本不一致导致的运行时崩溃。








