在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 之间的 "翻译官",让两者能正确识别对方编译的函数。

相关推荐
玉树临风ives2 分钟前
atcoder ABC 453 题解
数据结构·c++·算法·图论·atcoder
小则又沐风a3 分钟前
STL库: string类
开发语言·c++
△曉風殘月〆8 分钟前
一文带你掌握Visual Studio中集成的git功能
git·visual studio
mmz120712 分钟前
深度优先搜索DFS2(c++)
c++·算法·深度优先
6Hzlia12 分钟前
【Hot 100 刷题计划】 LeetCode 169. 多数元素 | C++ 哈希表基础解法
c++·leetcode·散列表
暴力求解15 分钟前
C++ ---string类(三)
开发语言·c++
itman30129 分钟前
C语言入门:掌握编程底层逻辑与核心技能
c语言·编程入门·系统开发·底层逻辑·核心技能
航Hang*31 分钟前
Windows Server 配置与管理——第7章:配置DNS服务器
运维·服务器·网络·windows·安全·虚拟化
say_fall37 分钟前
有关算法的简单数学问题
数据结构·c++·算法·职场和发展·蓝桥杯
꯭爿꯭巎꯭1 小时前
visual studio code (vscode)下载
ide·vscode·编辑器