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.rhkb.cn/news/446011.html

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

相关文章

MokeJs使用实例

文章目录 MokeJs使用实例介绍使用安装配置文件导入配置到main.js使用 axios 发送网络请求测试&#xff08;如果不会axios&#xff0c;具体可以见上篇文章axios&#xff09;启动示例 MokeJs使用实例 介绍 使用 安装 npm install mockjs --save-dev # 或者 yarn add mockj…

【超详细】基于YOLOv11的PCB缺陷检测

主要内容如下&#xff1a; 1、数据集介绍 2、下载PCB数据集 3、不同格式数据集预处理&#xff08;Json/xml&#xff09;&#xff0c;制作YOLO格式训练集 4、模型训练及可视化 5、Onnxruntime推理 运行环境&#xff1a;Python3.8&#xff08;要求>3.8&#xff09;&#xff…

matlab不小心删除怎么撤回

预设项——>删除文件——>移动至临时文件夹 tem临时文件夹下

【RabbitMQ】初识 RabbitMQ

&#x1f970;&#x1f970;&#x1f970;来都来了&#xff0c;不妨点个关注叭&#xff01; &#x1f449;博客主页&#xff1a;欢迎各位大佬!&#x1f448; 文章目录 1. MQ 是什么&#xff1f;1.1 MQ 本质1.2 系统间通信 2. MQ的作用是什么&#xff1f;2.1 异步解耦2.2 流量削…

【ProtoBuf】ProtoBuf基础与安装

本篇文章介绍 C 使用方向 文章目录 ProtoBuf简介ProtoBuf安装WindowsLinux ProtoBuf简介 ProtoBuf(全称为 Protocol Buffer)是一种序列化结构数据的方法 序列化是将对象转换为可存储的或传输的格式的过程&#xff0c;通常用于数据交换或持久化存储。我们在C/Java中编写的类不…

2.13寸电子墨水屏HINK-E0213+esp8266

记录好数字 2.13寸电子墨水屏HINK-E0213esp8266 声明:大部分资料来源于微雪电子http://微雪电子-官网 https://www.waveshare.net/ 前言 很久以前买的一块电子墨水屏,运气很好,这个型号HINK-E0213资料很全,而且微雪官网也有相关电路资料http://2.13inch e-Paper HAT - Waves…

【GaussDB】产品简介

产品定位 GaussDB 200是一款具备分析及混合负载能力的分布式数据库&#xff0c;支持x86和Kunpeng硬件架构&#xff0c;支持行存储与列存储&#xff0c;提供PB(Petabyte)级数据分析能力、多模分析能力和实时处理能力&#xff0c;用于数据仓库、数据集市、实时分析、实时决策和混…

【UI】将 naive ui 的 message 封装进axios 中,关于naiveui的message相关的用法

文章目录 前言在setup外进行使用直接包裹使用vue 单文件中使用 参考文章&#xff1a; 关于naiveui的message相关的用法 前言 最近新建了一个vite vu3 的项目&#xff0c;完全是从0 到1 &#xff0c;封装到request 的时候 想对axios 请求做一个全局的处理&#xff0c;但发现…

【尚硅谷】FreeRTOS学笔记(更新中更新时间2024.10.12)

在网上看到的一段很形象的描述&#xff0c;放在这里给大家娱乐一下。 裸机开发&#xff1a;n个人拉屎&#xff0c;先进去一个拉完&#xff0c;下一个再来。看门狗&#xff1a;如果有人拉完屎还占着&#xff0c;茅坑刷视频&#xff0c;把他拖出去中断系统&#xff1a;n个人拉屎&…

两三年没涨薪了

前几天到上海见合伙人&#xff0c;有好几位合伙人也都是中型或者是大厂的骨干&#xff0c;基本上是在P8这个级别&#xff0c;大家谈到了几个共同点。 几个典型的现象说一下&#xff0c;既是新闻&#xff0c;也是旧故事。天下的事都雷同。第一个&#xff0c;老板换了&#xff0c…

LVS-DR+Keepalived 高可用群集部署

LVS-DRKeepalived 高可用群集部署 Keepalived 的工作原理LVSKeepalived 高可用群集部署配置负载调度器&#xff08;主、备相同&#xff09;关闭防火墙和核心防护及准备IPVS模块配置keeplived&#xff08;主、备DR 服务器上都要设置&#xff09;启动 ipvsadm 服务调整 proc 响应…

机器学习中的模型设计与训练流程详解

目录 前言1. 模型设计1.1 数据特性分析1.2 计算资源限制1.3 应用场景需求 2. 模型训练2.1 训练集与验证集的划分2.2 损失函数的选择2.3 模型参数更新 3. 优化方法3.1 梯度下降法3.2 正则化方法 4. 模型测试4.1 性能评估指标4.2 模型的泛化能力 5. 模型选择5.1 数据规模与模型复…

集合框架06:Vector集合使用

1.视频链接&#xff1a;13.13 Vector使用_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1zD4y1Q7Fw/?p13&spm_id_from333.1007.top_right_bar_window_history.content.click&vd_sourceb5775c3a4ea16a5306db9c7c1c1486b52.代码示例&#xff1a; package com.yu…

Fetch 与 Axios:JavaScript HTTP 请求库的详细比较

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐&#xff1a;「storm…

初知C++:AVL树

文章目录 初知C&#xff1a;AVL树1.AVL树的概念2.AVL树的是实现2.1.AVL树的结构2.2.AVL树的插入2.3.旋转2.4.AVL树的查找2.5.AVL树平衡检测 初知C&#xff1a;AVL树 1.AVL树的概念 • AVL树是最先发明的自平衡⼆叉查找树&#xff0c;AVL是⼀颗空树&#xff0c;或者具备下列性…

python如何对变量赋值

Python 中的变量赋值不需要类型声明。 每个变量在内存中创建&#xff0c;都包括变量的标识&#xff0c;名称和数据这些信息。 每个变量在使用前都必须赋值&#xff0c;变量赋值以后该变量才会被创建。 等号&#xff08;&#xff09;用来给变量赋值。 等号&#xff08;&…

SpringBoot 整合 阿里云 OSS图片上传

一、OOS 简介 ‌阿里云OSS&#xff08;Object Storage Service&#xff09;是一种基于云存储的产品&#xff0c;适用于存储和管理各种类型的文件&#xff0c;包括图片、视频、文档等。‌ 阿里云OSS具有高可靠性、高可用性和低成本等优点&#xff0c;因此被广泛应用于各种场景&…

2013年国赛高教杯数学建模A题车道被占用对城市道路通行能力的影响解题全过程文档及程序

2013年国赛高教杯数学建模 A题 车道被占用对城市道路通行能力的影响 车道被占用是指因交通事故、路边停车、占道施工等因素&#xff0c;导致车道或道路横断面通行能力在单位时间内降低的现象。由于城市道路具有交通流密度大、连续性强等特点&#xff0c;一条车道被占用&#x…

el-image预览时和el-table边框出现样式穿透问题处理

el-image预览时和el-table边框出现样式穿透问题处理 如图所示 我们只需要在当前组件加一个css即可解决问题 <style lang"scss" scoped> :deep(.el-table__cell) {position: static !important; } </style>

【Golang】关于Go语言中的定时器原理与实战应用

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…