什么是C++模块化系统?C++20的模块化系统。
C++20引入的模块化系统是一种新的代码组织和编译机制,它旨在替代传统的头文件机制,提供更好的代码组织、更快的编译速度和更强的封装性。模块化系统的主要目标包括:
- 减少编译时间:通过减少冗余的头文件解析和宏定义传播,模块可以显著减少编译时间。
- 增强代码封装性:模块提供了更好的封装,减少了不必要的依赖暴露。
- 改善代码可维护性:模块可以更清晰地表达模块间的依赖关系,提高代码的可维护性。
传统头文件机制的问题
在传统的头文件机制中,代码往往通过#include指令包含其他文件的内容。这种方式存在一些问题:
- 重复解析:每次编译都会重新解析和处理包含的头文件,导致冗余的工作,特别是在大型项目中。
- 宏污染:头文件中的宏定义会在整个项目中传播,容易导致命名冲突和不一致。
- 编译时间长:由于重复的解析和处理,编译时间会随着项目规模的增加而显著增长。
模块化系统的工作方式
C++20的模块化系统通过引入模块声明和导入来替代传统的头文件机制。模块化系统的核心概念包括:
- 模块声明:定义模块的内容和接口。
- 模块导入:在需要使用模块的地方,明确地导入模块。
1. 模块声明(Module Declaration)
模块声明定义了一个模块,描述了模块的接口和实现。一个模块通常分为两个部分:
- 模块接口单元(Module Interface Unit):定义模块的外部接口,其他代码可以通过导入这个单元来使用模块的功能。
- 模块实现单元(Module Implementation Unit):包含模块的内部实现细节,不会直接暴露给模块的使用者。
模块声明的基本语法如下:
// module_name.ixx
export module module_name; // 定义模块接口单元export int add(int a, int b); // 导出模块接口
在这个例子中,module_name.ixx定义了一个名为module_name的模块,并导出一个名为add的函数。
2. 模块导入(Module Import)
模块导入允许其他代码使用模块中定义的接口。导入模块的基本语法如下:
// main.cpp
import module_name; // 导入模块int main() {int result = add(2, 3); // 使用导入的模块函数return 0;
}
在这个例子中,main.cpp导入了module_name模块,并使用了其中定义的add函数。
使用模块替代传统的头文件机制
1. 创建模块接口单元
模块接口单元包含了模块的公共接口,可以用来替代传统的头文件。在模块接口单元中,我们可以定义和导出模块的公共接口:
// math.ixx
export module math; // 定义模块export int add(int a, int b); // 导出函数
export int subtract(int a, int b);
2. 创建模块实现单元
模块实现单元包含了模块的具体实现细节,可以用来替代传统的源文件:
// math_impl.cpp
module math; // 声明模块的实现单元int add(int a, int b) {return a + b;
}int subtract(int a, int b) {return a - b;
}
在这里,math_impl.cpp是math模块的实现单元,它实现了add和subtract函数。
3. 导入和使用模块
在需要使用模块的地方,我们可以导入模块,而不需要包含头文件:
// main.cpp
import math; // 导入模块int main() {int result1 = add(10, 5); // 使用导入的模块函数int result2 = subtract(10, 5);return 0;
}
在这个例子中,我们导入了math模块,并使用了它提供的函数add和subtract。
模块化系统的优势
- 减少编译时间:模块只需要编译一次,并且编译后的模块可以被多个编译单元复用,从而减少整体编译时间。
- 增强封装性:模块明确规定了接口和实现的分离,不会像头文件那样暴露内部实现细节。
- 减少宏污染:模块内部的宏和定义不会泄露到模块外部,减少了命名冲突的风险。
- 改善代码组织:模块使得代码的依赖关系更加明确和清晰,增强了代码的可维护性。
实践中的模块使用
1. 组织大型项目
在大型项目中,可以将功能相近的代码组织成模块。例如,一个包含多种数学操作的库可以被组织成多个模块:
// arithmetic.ixx
export module arithmetic;export int add(int a, int b);
export int subtract(int a, int b);// geometry.ixx
export module geometry;export double area_of_circle(double radius);
export double perimeter_of_square(double side);
每个模块专注于特定的功能,便于代码的维护和扩展。
2. 与传统代码的兼容性
在迁移到模块化系统时,仍然可以保留一些传统的头文件,以便与不支持模块的旧代码或第三方库兼容。例如,可以将传统的头文件包裹在模块中:
// legacy_header.h
#ifndef LEGACY_HEADER_H
#define LEGACY_HEADER_Hvoid legacy_function();#endif // LEGACY_HEADER_H
// legacy_module.ixx
export module legacy_module;
#include "legacy_header.h"export using ::legacy_function; // 导出传统的函数
通过这种方式,可以逐步过渡到模块化系统,而不需要一次性重写所有代码。
编译和构建模块
为了编译和构建模块化的代码,编译器和构建系统需要支持C++20的模块特性。常见的编译器和构建系统通常都有相应的支持。例如,在GCC或Clang中,可以使用以下命令编译模块:
# 编译模块接口单元
g++ -std=c++20 -fmodules-ts -c math.ixx -o math.o# 编译模块实现单元
g++ -std=c++20 -fmodules-ts -c math_impl.cpp -o math_impl.o# 编译和链接主程序
g++ -std=c++20 -fmodules-ts main.cpp math.o math_impl.o -o main
在现代的构建系统(如CMake)中,也可以配置模块的构建规则,简化模块化代码的编译过程。
总结
C++20的模块化系统通过引入模块声明和导入,提供了一种替代传统头文件机制的现代化解决方案。模块化系统不仅减少了编译时间,增强了代码的封装性,还改善了代码的组织和可维护性。在实践中,模块化系统可以帮助我们更高效地管理和扩展大型项目,同时兼容传统的代码结构,使得迁移到新系统变得更加平滑。
相关文章:
什么是C++模块化系统?C++20的模块化系统。
C20引入的模块化系统是一种新的代码组织和编译机制,它旨在替代传统的头文件机制,提供更好的代码组织、更快的编译速度和更强的封装性。模块化系统的主要目标包括: 减少编译时间:通过减少冗余的头文件解析和宏定义传播,…...
智慧校园-档案管理系统总体概述
智慧校园档案管理系统,作为教育信息化进程中的重要一环,它运用现代信息技术的力量,彻底改变了传统档案管理的面貌,为学校档案资源的收集、整理、存储、检索与利用开辟了全新的途径。这一系统全面覆盖学生、教职工、教学科研及行政…...
文心一言 VS 讯飞星火 VS chatgpt (290)-- 算法导论21.3 3题
三、给出一个包含 m 个 MAKE-SET 、UNION 和 FINDSET 操作的序列(其中有 n 个是 MAKE-SET 操作),当仅使用按秩合并时,需要 Ω(mlgn) 的时间。如果要写代码,请用go语言。 文心一言: 在并查集(Union-Find)数…...
逻辑回归梯度推导
逻辑回归是一种广泛使用的分类算法,用于估计一个事件发生的概率。它是线性回归的扩展,通过sigmoid函数将线性回归的输出映射到[0, 1]区间,从而用于分类任务。 在逻辑回归中,我们使用对数似然损失函数(log-likelihood l…...
Python 使用函数输出一个整数的逆序数
在Python中,你可以定义一个函数来输出一个整数的逆序数。这里有一个简单的实现方法: def reverse_integer(x):# 检查输入是否为整数if not isinstance(x, int):raise ValueError("Input must be an integer")# 将整数转换为字符串,…...
【Linux】Wmware Esxi磁盘扩容
目录 一、概述 1.1 磁盘分区概念 1.2 LVM概念 二、扩容步骤 二、报错 一、概述 1.1 磁盘分区概念 在 Linux 中,每一个硬件设备都映射到一个系统的文件,对于硬盘、光驱等 IDE 或 SCSI 设备也不例外。Linux把各种 IDE 设备分配了一个由 hd 前缀组成的文…...
树莓派4B_OpenCv学习笔记15:OpenCv定位物体实时坐标
今日继续学习树莓派4B 4G:(Raspberry Pi,简称RPi或RasPi) 本人所用树莓派4B 装载的系统与版本如下: 版本可用命令 (lsb_release -a) 查询: Opencv 版本是4.5.1: 今日学习 OpenCv定位物体实时位置,代码来源是…...
MySQL之如何定位慢查询
1、如何定位慢查询 1.1、使用开源工具 调试工具:Arthas 运维工具:Promethuss、Skywalking 1.2、MySQL自带慢日志 慢查询日志记录了所有执行时间超过指定参数(long_query_time,单位:秒,默认10秒&#x…...
Open3D 删除点云中重复的点
目录 一、算法原理1、重叠点2、主要函数二、代码实现三、结果展示本文由CSDN点云侠原创,原文链接。如果你不是在点云侠的博客中看到该文章,那么此处便是不要脸的爬虫与GPT。 一、算法原理 1、重叠点 原始点云克隆一份 构造重叠区域 合并点云获得重叠点 2、主要…...
填报志愿选专业是兴趣重要还是前景重要?
进行专业评估,找到一个适合自己的专业是一件非常困难的事情。在进行专业选择时,身上理想化色彩非常严重的人,会全然不顾及他人的劝阻,义无反顾的以兴趣为主,选择自己热爱的专业。一些较多考虑他人建议,能听…...
python开发基础——day9 函数基础与函数参数
一、初识函数(function) 编程函数!数学函数,里面的是逻辑,功能,而不是套公式 编程函数的作用实现特定操作的一段代码 你现在请客,每个人都点同样的一份吃的,请100个人 1.薯条 2.上校鸡块 3.可乐 那…...
STM32——使用TIM输出比较产生PWM波形控制舵机转角
一、输出比较简介: 只有高级定时器和通用寄存器才有输入捕获/输出比较电路,他们有四个CCR(捕获/比较寄存器),共用一个CNT(计数器),而输出比较功能是用来输出PWM波形的。 红圈部分…...
第十五章 集合(set)(Python)
文章目录 前言一、集合 前言 集合(set)是一个无序的不重复元素序列。 一、集合 set {1, 2, 3, 4}...
面试-javaIO机制
1.BIO BIO:是传统的javaIO以及部分java.net下部分接口和类。例如,socket,http等,因为网络通信同样是IO行为。传统IO基于字节流和字符流进行操作。提供了我们最熟悉的IO功能,譬如基于字节流的InputStream 和OutputStream.基于字符流…...
在.NET Core中,config和ConfigureServices的区别和作用
在.NET Core中,config和ConfigureServices是两个不同的概念,它们在应用程序的启动和配置过程中扮演着不同的角色。 ConfigureServices:这是ASP.NET Core应用程序中的一个方法,位于Startup类的内部。它的作用是配置依赖注入(DI)容器…...
App Inventor 2 如何实现多个定时功能?
1、可以使用多个“计时器”组件。 2、也可以用一个计时器,定时一分钟。也就是一分钟就会触发一次事件执行,定义一个全局数字变量,在事件中递增,用逻辑判断这个变量的值即可完成多个想要定时的任务(о∀о) 代码块请参考…...
技术驱动的音乐变革:AI带来的产业重塑
📑引言 近一个月来,随着几款音乐大模型的轮番上线,AI在音乐产业的角色迅速扩大。这些模型不仅将音乐创作的门槛降至前所未有的低点,还引发了一场关于AI是否会彻底颠覆音乐行业的激烈讨论。从初期的兴奋到现在的理性审视࿰…...
重生之我要学后端0--HTTP协议和RESTful APIs
http和RESTful APIs HTTP协议RESTful APIs设计RESTful API设计实例 HTTP协议 HTTP(超文本传输协议)是用于分布式、协作式和超媒体信息系统的应用层协议。它是网页数据通讯的基础。工作原理简述如下: 客户端请求(Request…...
深度之眼(二十八)——神经网络基础知识(三)-卷积神经网络
文章目录 一、前言二、卷积操作2.1 填充(padding)2.2 步长2.3 输出特征图尺寸计算2.4 多通道卷积 三、池化操作四、Lenet-5及CNN结构进化史4.1 Lenet-5 一、前言 卷积神经网络–AlexNet(最牛)-2012 Lenet-5-大规模商用(1989) 二、…...
AI Infra简单记录
向量数据库的作用 1. 在AI大模型训练过程中,向量数据库可以有效提升数据检索、特征提取等任务的效率。 2、在AI大模型推理过程中,向量数据库为大模型提供外挂知识库,提升模型时效性与准确性,提供缓存能力,减少调用开…...
理解 MCP 工作流:使用 Ollama 和 LangChain 构建本地 MCP 客户端
🌟 什么是 MCP? 模型控制协议 (MCP) 是一种创新的协议,旨在无缝连接 AI 模型与应用程序。 MCP 是一个开源协议,它标准化了我们的 LLM 应用程序连接所需工具和数据源并与之协作的方式。 可以把它想象成你的 AI 模型 和想要使用它…...
OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别
OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别 直接训练提示词嵌入向量的核心区别 您提到的代码: prompt_embedding = initial_embedding.clone().requires_grad_(True) optimizer = torch.optim.Adam([prompt_embedding...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据
微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列,以便知晓哪些列包含有价值的数据,…...
以光量子为例,详解量子获取方式
光量子技术获取量子比特可在室温下进行。该方式有望通过与名为硅光子学(silicon photonics)的光波导(optical waveguide)芯片制造技术和光纤等光通信技术相结合来实现量子计算机。量子力学中,光既是波又是粒子。光子本…...
【网络安全】开源系统getshell漏洞挖掘
审计过程: 在入口文件admin/index.php中: 用户可以通过m,c,a等参数控制加载的文件和方法,在app/system/entrance.php中存在重点代码: 当M_TYPE system并且M_MODULE include时,会设置常量PATH_OWN_FILE为PATH_APP.M_T…...
WPF八大法则:告别模态窗口卡顿
⚙️ 核心问题:阻塞式模态窗口的缺陷 原始代码中ShowDialog()会阻塞UI线程,导致后续逻辑无法执行: var result modalWindow.ShowDialog(); // 线程阻塞 ProcessResult(result); // 必须等待窗口关闭根本问题:…...
华为OD最新机试真题-数组组成的最小数字-OD统一考试(B卷)
题目描述 给定一个整型数组,请从该数组中选择3个元素 组成最小数字并输出 (如果数组长度小于3,则选择数组中所有元素来组成最小数字)。 输入描述 行用半角逗号分割的字符串记录的整型数组,0<数组长度<= 100,0<整数的取值范围<= 10000。 输出描述 由3个元素组成…...
MySQL的pymysql操作
本章是MySQL的最后一章,MySQL到此完结,下一站Hadoop!!! 这章很简单,完整代码在最后,详细讲解之前python课程里面也有,感兴趣的可以往前找一下 一、查询操作 我们需要打开pycharm …...
土建施工员考试:建筑施工技术重点知识有哪些?
《管理实务》是土建施工员考试中侧重实操应用与管理能力的科目,核心考查施工组织、质量安全、进度成本等现场管理要点。以下是结合考试大纲与高频考点整理的重点内容,附学习方向和应试技巧: 一、施工组织与进度管理 核心目标: 规…...
