在C++ 程序中调用被 C编译器编译后的函数,为什么要加 extern “C”声明?

一、先搞懂核心问题:C 和 C++ 的 "函数名编译差异"

C++ 支持函数重载 (比如func(int)func(float)可以共存),为了区分不同重载的函数,C++ 编译器会对函数名做「名字修饰(Name Mangling)」------ 把函数的参数类型、个数等信息拼接到函数名里,生成一个独一无二的编译后名字。而 C 语言不支持重载,C 编译器对函数名的处理非常简单:基本直接保留原函数名(或仅做简单修饰)。

举个直观例子:

定义一个简单函数 void func(int);

  • C 编译器编译后 :函数名还是func(或类似_func的简单形式);
  • C++ 编译器编译后 :会变成_Z4funci(不同编译器格式略有差异,核心是拼接了参数类型int的标识)。

二、不加 extern "C" 会发生什么?

假设你有一个用 C 编译器编译的函数库,里面有void func(int);,编译后函数名是func;当你在 C++ 代码里直接调用func(10);时:

  1. C++ 编译器会按自己的规则,把代码里的func解析成修饰后的名字(比如_Z4funci);
  2. 链接阶段,编译器会去找_Z4funci这个函数,但 C 编译的库中只有func,找不到对应符号,最终报链接错误(undefined reference to func(int)

三、extern "C" 的作用:告诉 C++ 编译器 "按 C 规则处理"

extern "C"是 C++ 的特有语法,它的核心作用是:让 C++ 编译器对被修饰的函数,按 C 语言的规则处理(包括名字修饰、调用约定等)

正确的使用方式(示例):
cpp 复制代码
// 1. 声明C编译的函数(告诉C++按C规则找这个函数)
extern "C" {
    void func(int); // 这个函数是C编译器编译的
}

// 2. 调用函数(C++编译器会按C的名字"func"去链接)
int main() {
    func(10); 
    return 0;
}

如果是批量引入 C 的头文件,更常用的写法(兼顾 C/C++ 编译):

cpp 复制代码
// 假设这是C的头文件 func.h
#ifdef __cplusplus // 识别C++编译器
extern "C" {
#endif

void func(int); // C风格函数声明

#ifdef __cplusplus
}
#endif

四、补充:为什么 C 不需要处理 C++ 的函数?

因为 C 语言不支持extern "C"语法,也不会去调用 C++ 的重载函数 ------ 通常都是 C++ 调用 C 的代码(比如老的 C 语言库),所以只需要在 C++ 侧做声明即可。

总结

  1. 核心原因 :C++ 的「名字修饰」和 C 不同,不加extern "C"会导致链接时找不到 C 编译的函数;
  2. extern "C" 的作用:强制 C++ 编译器对目标函数采用 C 的名字修饰规则,匹配 C 编译的函数符号;
  3. 使用场景:C++ 调用 C 语言编写 / 编译的函数、库(如 C 标准库、第三方 C 库)时必须加。

简单记:extern "C"就是 C++ 和 C 之间的 "翻译官",让两者能正确识别对方编译的函数。

相关推荐
计算机安禾6 分钟前
【C语言程序设计】第35篇:文件的打开、关闭与读写操作
c语言·开发语言·c++·vscode·算法·visual studio code·visual studio
xiaoliuliu1234519 分钟前
Autodesk官方卸载工具使用教程(Windows版,含解压+管理员运行+批量卸载)
windows
不想写代码的星星25 分钟前
告别 C 风格枚举:为什么你应该使用 enum class
c++
m0_7336122129 分钟前
C++20概念(Concepts)入门指南
开发语言·c++·算法
AI_搬运工29 分钟前
从智能指针窥见现代C++的生存法则:告别内存泄漏,这篇就够了
c++
仰泳的熊猫31 分钟前
题目2571:蓝桥杯2020年第十一届省赛真题-回文日期
数据结构·c++·算法·蓝桥杯
CODE_RabbitV40 分钟前
【3min 解决】keil5 编译stm32 出现一堆 core_cm3.c 报错问题
c语言·stm32·嵌入式硬件
2301_807367191 小时前
C++中的模板方法模式
开发语言·c++·算法
weixin_537590451 小时前
《C程序语言设计》练习答案(练习1-3)
c语言·开发语言
sdm0704271 小时前
基础开发工具git,gdb
git