extern “C“ 的作用、C++ 和 C 编译的不同、C++ 编译过程的五个主要阶段

在 C++ 中,如果需要从 C 语言导入函数或与 C 代码交互,需要使用 extern "C" 关键字。这是因为 C++ 和 C 在编译过程中的 符号命名机制(即 "名称修饰" 或 "name mangling")不同。

1. extern "C" 的作用

extern "C" 关键字的主要作用是告诉 C++ 编译器,所包含的函数或变量使用的是 C 语言的命名和链接规则,而不是 C++ 的。这对于编译和链接 C++ 代码与 C 代码特别重要,因为 C++ 编译器在编译函数时会进行名称修饰,而 C 编译器不会。

C++ 名称修饰:C++ 支持函数重载、类成员函数等复杂的功能,因此 C++ 编译器会对函数名进行编码,以区分同名的不同函数。这个过程称为 名称修饰(name mangling)。

而 C 语言不支持函数重载,其编译器不会对函数名进行修饰,保持函数名的简单性。因此,为了在 C++ 中使用 C 编写的函数,必须使用 extern "C" 来告知 C++ 编译器使用 C 语言的链接规则。

2. extern "C" 的用法

导入单个 C 函数

extern "C" {
    void my_c_function(int a);
}

这段代码告诉 C++ 编译器,my_c_function 函数是用 C 语言实现的,并且在编译时应遵循 C 语言的符号命名规则。

导入多个 C 函数

可以将多个函数的声明包含在 extern "C" 块中:

extern "C" {
    void my_c_function1(int a);
    int my_c_function2(double b);
}

导入 C 头文件

如果你想在 C++ 中包含一个由 C 编写的头文件,可以使用 extern "C" 包装整个头文件的内容:

extern "C" {
    #include "my_c_header.h"
}

或者,头文件本身可以被修改,使用条件编译保证其既可以在 C++ 中使用,也可以在 C 中使用:

#ifdef __cplusplus
extern "C" {
#endif

// C 函数声明
void my_c_function(int a);

#ifdef __cplusplus
}
#endif

在这种情况下,当头文件在 C++ 中被包含时,extern "C" 生效,而在 C 中编译时则不受影响。

3. C++ 和 C 编译的不同

3.1. 名称修饰(Name Mangling)

如前所述,C++ 编译器会对函数名进行名称修饰,以支持函数重载、命名空间和类成员函数等特性。名称修饰使得同一个函数名可以根据其参数、命名空间等特征进行区分。C 编译器则不做名称修饰,所有的函数名在编译后保持原样。

C++ 名称修饰示例

在 C++ 中,编译器可能会将函数 int my_function(int) 的名字修饰成 _Z11my_functioni,其中包括函数名和参数类型的编码。

C 函数名示例

在 C 语言中,函数 int my_function(int) 仍然保持为简单的 my_function

名称修饰的影响:
  • C++ 中的函数重载:C++ 支持函数重载,因此会为每个重载的函数生成不同的修饰名称。
  • C 函数命名:由于 C 不支持函数重载,所有函数的名称在编译时保持唯一,编译器不需要进行名称修饰。
3.2. 链接规则
  • C 链接:C 编译器在链接时只需要处理简单的函数名称。
  • C++ 链接:C++ 编译器在链接时需要处理名称修饰后的符号,因而 C++ 函数可以根据参数、类、命名空间等进行链接。
3.3. 语言特性
  • C++ 编译器:处理复杂的 C++ 语言特性,如类、模板、命名空间、异常处理、虚函数等。C++ 代码在编译时往往比 C 代码复杂得多。
  • C 编译器:处理 C 语言的相对简单的语法和功能,不需要考虑面向对象的特性。

4. extern "C" 的使用场景

  • C++ 调用 C 函数:当你在 C++ 代码中需要使用 C 库时,必须通过 extern "C" 来确保 C++ 编译器按照 C 的方式处理函数的链接。典型场景包括使用 C 编写的系统库、第三方库(如 POSIX、OpenSSL 等)。
  • C++ 提供 C 接口:如果你编写了 C++ 库,且需要提供给 C 语言使用,那么可以通过 extern "C" 来导出 C 风格的接口,使 C 语言能够调用 C++ 编译的库。
示例:C++ 调用 C 函数

假设有一个 C 编写的函数 my_c_function

// my_c_code.c
#include <stdio.h>

void my_c_function(int a) {
    printf("Value: %d\n", a);
}

在 C++ 中调用这个 C 函数时,需要使用 extern "C"

// my_cpp_code.cpp
extern "C" {
    void my_c_function(int a);
}

int main() {
    my_c_function(10);  // 调用C函数
    return 0;
}

示例:C++ 提供 C 接口

// my_cpp_code.cpp
extern "C" {
    void my_cpp_function(int a);
}

void my_cpp_function(int a) {
    // C++ 实现
    std::cout << "C++ Function: " << a << std::endl;
}

这个函数可以被 C 代码调用,因为它的符号遵循 C 的规则。

总结

  1. extern "C":用于告诉 C++ 编译器按照 C 的方式处理函数的链接,使 C++ 和 C 代码能够互相调用。
  2. C++ 编译和 C 编译的不同
    • 名称修饰:C++ 编译器会进行名称修饰,以支持函数重载和其他高级功能,而 C 编译器不会。
    • 链接规则:C++ 链接时处理更复杂的符号,而 C 语言保持简单的函数名。
  3. 常见使用场景
    • C++ 调用 C 库时需要 extern "C"
    • 提供 C++ 库接口给 C 语言使用时,也需要使用 extern "C"

C++ 程序从编写代码到生成可执行的二进制文件,通常经历以下 五个步骤,即:编写代码预处理编译汇编链接。每个步骤都会转换或处理代码,最终生成一个可执行文件。

C++ 编译过程的五个主要阶段

  1. 编写代码(Source Code Writing)
  2. 预处理(Preprocessing)
  3. 编译(Compilation)
  4. 汇编(Assembly)
  5. 链接(Linking)

1. 编写代码(Source Code Writing)

这是程序员编写的 C++ 源代码,通常使用文件扩展名 .cpp.h。C++ 源文件可能包含类、函数、数据结构和其他逻辑。

  • 源文件:例如 main.cpp, MyClass.cpp, MyClass.h

2. 预处理(Preprocessing)

预处理是编译的第一个阶段,C++ 编译器中的 预处理器 会处理所有以 # 开头的指令,如 #include, #define 等。这一步的主要任务是处理宏替换、文件包含和条件编译指令。

主要任务

  • 文件包含:将 #include 的头文件内容插入源文件。
  • 宏替换:将所有宏定义(通过 #define 定义的内容)替换为对应的值。
  • 条件编译:根据条件指令如 #ifdef#endif 等执行代码的保留或忽略。
  • 注释删除:删除所有的注释内容,包括单行注释 // 和多行注释 /* ... */

预处理后的文件仍然是文本文件,通常称为 扩展文件,但它包含了所有展开的宏、包含的头文件等。

预处理示例

#include <iostream>
#define MAX 100

int main() {
    std::cout << MAX << std::endl;
    return 0;
}

预处理后变成:

// 假设 <iostream> 已被展开
// 省略头文件展开的内容...

int main() {
    std::cout << 100 << std::endl;  // 宏 MAX 被替换为 100
    return 0;
}

3. 编译(Compilation)

在这一步,编译器 将预处理后的源代码文件(仍然是人类可读的 C++ 代码)转换为 中间代码,通常称为 汇编代码。汇编代码是面向特定硬件架构的一种低级别语言,但仍然是人类可读的符号代码。

编译的过程

  • 语法分析:编译器对源代码进行语法分析,确保代码符合 C++ 的语法规则。
  • 语义分析:检查代码的逻辑是否合理,比如类型检查、变量是否已声明等。
  • 生成汇编代码:编译器根据目标架构将 C++ 代码转换为汇编语言。

编译后的文件:通常以 .s 为扩展名,它仍然是人类可读的汇编代码。

编译示例

int main() {
    int a = 10;
    return a;
}

编译后的汇编代码(示例):

    .section    __TEXT,__text,regular,pure_instructions
    .globl  _main
_main:                                  ## @main
    .cfi_startproc
## %bb.0:
    pushq   %rbp
    .cfi_def_cfa_offset 16
    .cfi_offset %rbp, -16
    movq    %rsp, %rbp
    movl    $10, %eax
    popq    %rbp
    retq
    .cfi_endproc

4. 汇编(Assembly)

汇编器(Assembler)将编译生成的汇编代码转换为机器码(Machine Code),也称为 目标文件(Object File)。目标文件是特定于目标机器的二进制文件,但还不是完整的可执行文件,因为它可能仍然有外部依赖(比如其他模块中的函数)。

汇编的过程

  • 将汇编代码转换为机器指令,这些指令是计算机 CPU 可以直接理解并执行的二进制代码。

目标文件

  • 汇编器生成的目标文件通常具有 .o(Unix/Linux/Mac 系统)或 .obj(Windows 系统)扩展名。
  • 每个 C++ 源文件都会生成一个对应的目标文件。

示例: 一个目标文件内部包含计算机能执行的机器代码(无法直接展示二进制内容)。

5. 链接(Linking)

链接器(Linker)将多个目标文件和库文件组合起来,生成最终的可执行文件。链接器的主要任务是解决符号引用,即将函数调用和全局变量的引用链接到其实际定义上。

链接的过程

  • 符号解析:链接器会将各个目标文件中的未解析的符号(如函数、全局变量等)与其他目标文件或库中定义的符号进行匹配。
  • 库链接:如果程序使用了外部库(如标准库、动态库或静态库),链接器会将这些库与目标文件链接起来。
  • 生成可执行文件:链接器最终生成一个完整的二进制可执行文件,该文件可以直接在目标平台上运行。

链接示例

g++ main.o MyClass.o -o my_program

这条命令会将 main.oMyClass.o 链接在一起,生成可执行文件 my_program

6. 可执行文件(Executable File)

生成的可执行文件是二进制文件,包含了最终的机器代码以及所有的依赖库和符号。这个文件可以在目标系统上直接运行。

  • LinuxMac 系统的可执行文件没有特定的扩展名,通常通过运行 ./my_program 执行。
  • Windows 系统上的可执行文件通常有 .exe 扩展名,可以通过双击或命令行运行。

可视化编译过程

编译过程中的工具

  • 预处理器:负责处理 # 开头的预处理指令(如 #include, #define)。
  • 编译器:将 C++ 代码翻译为汇编代码(如 g++clang)。
  • 汇编器:将汇编代码转为机器码,生成目标文件(如 as)。
  • 链接器:负责将目标文件链接成可执行文件(如 ld)。

编译选项

在编译时可以使用一些编译器选项来控制不同的阶段。例如,使用 GCC 时:

  • -E:只执行预处理,不编译。
  • -S:将 C++ 代码编译成汇编代码,生成 .s 文件。
  • -c:只编译,不链接,生成目标文件 .o
  • -o:指定生成的可执行文件的名称。

总结

C++ 程序从代码到可执行文件的过程是通过预处理、编译、汇编和链接这几个步骤完成的。每个阶段都有特定的任务,最终生成可以在目标平台上运行的二进制文件。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/54957.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2014年国赛高教杯数学建模D题储药柜的设计解题全过程文档及程序

2014年国赛高教杯数学建模 D题 储药柜的设计 储药柜的结构类似于书橱&#xff0c;通常由若干个横向隔板和竖向隔板将储药柜分割成若干个储药槽(如图1所示)。为保证药品分拣的准确率&#xff0c;防止发药错误&#xff0c;一个储药槽内只能摆放同一种药品。药品在储药槽中的排列…

【机器学习】金融预测 —— 风险管理与股市预测

我的主页&#xff1a;2的n次方_ 在金融领域&#xff0c;机器学习&#xff08;ML&#xff09;已经成为了不可或缺的工具。金融预测&#xff0c;尤其是风险管理和股市预测&#xff0c;涉及海量数据和复杂模式的分析&#xff0c;而这些正是机器学习擅长处理的领域。通过分析历…

Sentinel最全笔记,详细使用步骤教程清单

一、Sentinel的基本功能 1、流量控制 流量控制在网络传输中是一个常用的概念&#xff0c;它用于调整网络包的发送数据。然而&#xff0c;从系统稳定性角度考虑&#xff0c;在处理请求的速度上&#xff0c;也有非常多的讲究。任意时间到来的请求往往是随机不可控的&#xff0c;…

[单master节点k8s部署]37.微服务(一)springCloud 微服务

微服务架构的一个重要特点是&#xff0c;它与开发中使用的具体编程语言或技术栈无关。每个微服务都可以使用最适合其功能需求的语言或技术来实现。例如&#xff0c;一个微服务可以用Java编写&#xff0c;另一个微服务可以用Python、Go、Node.js等编写。微服务架构允许这种灵活性…

数据结构-C语言顺序栈功能实现

栈 栈&#xff1a;类似于一个容器&#xff0c;如我们生活中的箱子&#xff0c;我们向箱子里放东西&#xff0c;那么最先放的东西是最后才能拿出来的 代码实现 #include <stdio.h> #include <stdlib.h>#define MAX_SIZE 100typedef struct {int* base; // 栈底指针…

通过PyTorch 手写数字识别 入门神经网络 详细讲解

通过PyTorch 手写数字识别 入门神经网络 数据集 MNIST数据集中有手写数字图片7万张&#xff0c;划分训练集6万张&#xff0c;划分测试集1万张。 每张图片都会有一张标签&#xff0c;也就是代表着图片的真实值&#xff08;真实含义&#xff09;。 概念 计算机是如何读取图片的…

多态常见面试问题

1、什么是多态&#xff1f; 多态&#xff08;Polymorphism&#xff09;是面向对象编程中的一个重要概念&#xff0c;它允许同一个接口表现出不同的行为。在C中&#xff0c;多态性主要通过虚函数来实现&#xff0c;分为编译时多态&#xff08;静态多态&#xff09;和运行时多态…

【Spring AI】Java实现类似langchain的第三方函数调用_原理与详细示例

Spring AI 介绍 &#xff1a;简化Java AI开发的统一接口解决方案 在过去&#xff0c;使用Java开发AI应用时面临的主要困境是没有统一且标准的封装库&#xff0c;导致开发者需要针对不同的AI服务提供商分别学习和对接各自的API&#xff0c;这增加了开发难度与迁移成本。而Sprin…

【数据结构】邻接表

一、概念 邻接表是一个顺序存储与链式存储相结合的数据结构&#xff0c;用于描述一个图中所有节点之间的关系。 若是一个稠密图&#xff0c;我们可以选择使用邻接矩阵&#xff1b;但当图较稀疏时&#xff0c;邻接矩阵就显得比较浪费空间了&#xff0c;此时我们就可以换成邻接…

机器人的应用 基于5G的变电站智慧管控系统

背景概述 一、电力行业面临的挑战与变革 随着全球工业化和信息化的快速发展&#xff0c;电力行业作为国民经济的基础性行业&#xff0c;其重要性日益凸显。然而&#xff0c;随着电力网络的不断扩展和复杂化&#xff0c;变电站和开关站作为电力传输与分配的关键节点&#xff0…

Excel中Ctrl+e的用法

重点&#xff1a;想要使用ctrle&#xff0c;前提是整合或拆分后的结果放置的单元格必须和被提取信息的单元格相邻&#xff0c;且被提取信息的单元格也必须相连。 下图为错误示例 这样则可以使用ctrle 1、信息整合 2、提取信息 3、添加符号 4、信息顺序调换 5、数字提取 crtle还…

HarmonyOS NEXT 应用开发实战(三、ArkUI页面底部导航TabBar的实现)

在开发HarmonyOS NEXT应用时&#xff0c;TabBar是用户界面设计中不可或缺的一部分。本文将通过代码示例&#xff0c;带领大家一同实现一个常用的TabBar&#xff0c;涵盖三个主要的内容页&#xff1a;首页、知乎日报和我的页面。以模仿知乎日报的项目为背景驱动&#xff0c;设定…

解决ubuntu 下 VS code 无法打开点击没反应问题

从Ubuntu 22.04 升级到ubuntu 24.04 后&#xff0c;发现Vsode无法打开&#xff0c;不论是点击图标&#xff0c;还是terminator里面运行code 可执行程序&#xff0c;均没有反应。debug如下: 提示权限不够。 解决方案&#xff1a; sudo sysctl -w kernel.apparmor_restrict_unp…

C语言题目练习2

前面我们知道了单链表的结构及其一些数据操作&#xff0c;今天我们来看看有关于单链表的题目~ 移除链表元素 移除链表元素&#xff1a; https://leetcode.cn/problems/remove-linked-list-elements/description/ 这个题目要求我们删除链表中是指定数据的结点&#xff0c;最终返…

C语言 | Leetcode C语言题解之第460题LFU缓存

题目&#xff1a; 题解&#xff1a; /* 数值链表的节点定义。 */ typedef struct ValueListNode_s {int key;int value;int counter;struct ValueListNode_s *prev;struct ValueListNode_s *next; } ValueListNode;/* 计数链表的节点定义。 其中&#xff0c;head是数值链表的头…

腾讯云Android 与 iOS 相关

移动端&#xff08;Android/iOS&#xff09;支持哪几种系统音量模式&#xff1f; 支持2种系统音量类型&#xff0c;即通话音量类型和媒体音量类型&#xff1a; 通话音量&#xff1a;手机专门为通话场景设计的音量类型&#xff0c;使用手机自带的回声抵消功能&#xff0c;音质…

谷歌浏览器 文件下载提示网络错误

情况描述&#xff1a; 谷歌版本&#xff1a;129.0.6668.90 (正式版本) &#xff08;64 位&#xff09; (cohort: Control)其他浏览器&#xff0c;比如火狐没有问题&#xff0c;但是谷歌会下载失败&#xff0c;故推断为谷歌浏览器导致的问题小文件比如1、2M会成功&#xff0c;大…

【LeetCode】动态规划—95. 不同的二叉搜索树 II(附完整Python/C++代码)

动态规划—95. 不同的二叉搜索树 II 题目描述前言基本思路1. 问题定义二叉搜索树的性质&#xff1a; 2. 理解问题和递推关系递归构造思想&#xff1a;状态定义&#xff1a;递推公式&#xff1a;终止条件&#xff1a; 3. 解决方法递归 动态规划方法&#xff1a;伪代码&#xff…

如何使用vscode的launch.json来debug调试

1、创建一个launch.json文件 选择Python Debugger&#xff0c;再选择Python文件&#xff0c;创建处理如下 默认有下面五个参数 "name": "Python Debugger: Current File","type": "debugpy","request": "launch"…

金九银十软件测试面试题(800道)

今年你的目标是拿下大厂offer&#xff1f;还是多少万年薪&#xff1f;其实这些都离不开日积月累的过程。 为此我特意整理出一份&#xff08;超详细笔记/面试题&#xff09;它几乎涵盖了所有的测试开发技术栈&#xff0c;非常珍贵&#xff0c;人手一份 肝完进大厂 妥妥的&#…