C++20 模块化(Modules)

news/2024/9/20 11:31:29/ 标签: c++

C++20 引入的模块化(Modules)是一个重大改进,旨在取代传统的头文件机制,提高编译速度、代码可维护性以及项目的可扩展性。模块化为 C++ 提供了一种更现代化的代码组织方式,避免了头文件中常见的宏污染、重复编译和复杂的依赖管理问题。

概念与背景

在 C++20 之前,C++ 项目是通过头文件.h.hpp)与源文件.cpp)的组合来组织代码的。头文件定义了类、函数、模板等声明,而源文件包含它们的实现。然而,头文件机制存在多个问题:

  • 重复编译:每次编译器遇到 #include 指令时,都会重新处理整个头文件。
  • 编译时间长:大型项目中,成千上万的头文件包含会导致严重的编译性能问题。
  • 符号冲突:头文件中的宏可能会与其他代码发生冲突,产生难以调试的错误。

模块化解决了这些问题,提供了更清晰的代码分隔、减少重复编译并减少符号冲突的风险。

模块的基本概念

1. 什么是模块?

模块是一个新的 C++ 编译单元,它由一个或多个模块文件组成,定义了导出(export)或隐藏的声明。模块打破了传统的头文件/源文件模式,允许开发者显式地控制哪些声明可以被外部使用。

模块的特点:

  • 明确的接口和实现:模块可以显式导出哪些部分是公共接口,哪些部分是私有实现。
  • 无宏污染:模块之间的依赖不会通过宏或预处理器泄漏。
  • 增量编译:模块化加快了增量编译的速度,减少了重复解析。
2. 模块文件结构

模块主要包括两种文件:

  • 模块接口文件(Module Interface File):包含导出给其他模块或翻译单元使用的声明。通常使用 .cppm 扩展名。
  • 模块实现文件(Module Implementation File):包含模块内部实现,通常以 .cpp 形式存在。

模块的语法和使用

1. 创建一个模块

模块通过 module 关键字定义模块名称。模块接口文件通常是 .cppm 文件,但这不是强制的。

// math.cppm - 模块接口文件
export module math;export int add(int a, int b) {return a + b;
}
  • export module 定义了模块的名称 math
  • export 关键字用于导出 add 函数,使其可供其他模块或翻译单元使用。
2. 使用模块

模块通过 import 关键字来导入,而不是像传统头文件那样使用 #include

// main.cpp - 使用 math 模块
import math;int main() {int result = add(2, 3);return 0;
}

与头文件不同,import 语句会将模块编译为二进制形式,避免了重复编译带来的性能损失。

3. 模块中的隐藏实现

模块允许将某些实现隐藏,而不导出给外部使用。只有 export 导出的内容才能被其他模块或翻译单元访问。

// math.cppm
export module math;export int add(int a, int b);  // 导出的函数声明int multiply(int a, int b) {   // 私有函数,不导出return a * b;
}export int add(int a, int b) {return a + b;
}

在这个例子中,add 函数是导出的,而 multiply 函数则是私有的,只能在模块内部使用。

模块化的更多细节

1. 模块分区(Module Partitioning)

C++20 允许将模块分成多个子模块,称为模块分区(Module Partitions)。模块分区帮助大型模块进行更细粒度的代码组织。

// math.cppm - 模块接口文件
export module math;
export import :operations;  // 导入子模块export int add(int a, int b);// operations.cppm - 模块分区文件
module math:operations;int multiply(int a, int b) {return a * b;
}

在这个例子中,模块 math 分为两个部分,主模块和 operations 子模块。主模块导入了子模块并公开了它的接口。

2. 模块与传统代码的混合使用

模块化的引入并不意味着完全抛弃传统的头文件机制。在模块化过渡期,模块与头文件可以共存。模块化代码仍然可以包含头文件,并与 #include 共用。

// math.cppm
export module math;
#include <iostream>  // 模块中仍可以使用头文件export void print_message() {std::cout << "Hello from module" << std::endl;
}
3. 模块的依赖管理

模块通过 import 显式管理依赖关系,而不是隐式地通过 #include 传播依赖。这种显式依赖管理使得模块的依赖关系更加清晰,并且减少了不必要的编译。

// math.cppm
export module math;
import <iostream>;  // 导入标准库模块export void print_message() {std::cout << "Message from math module" << std::endl;
}
4. 标准库模块化

C++20 模块化支持标准库模块。例如,标准库可以通过 import <iostream> 的形式来使用。

import <iostream>;int main() {std::cout << "Hello, Modules!" << std::endl;
}

 


http://www.ppmy.cn/news/1527972.html

相关文章

TESSY创建需要高级桩的测试用例

需要打高级桩的情况如下&#xff1a; 1) 使用到桩函数的返回值&#xff1b; 2) 如果函数有形参&#xff0c;并且需要接口传参检测&#xff1b; 我们以tessy5.1 IDE为例&#xff0c;给大家展示编写一个需要高级桩的测试用例过程。 1、前期的准备工作 可以参考以下文章&…

【Linux】yum、vim、gcc使用(超详细)

Linux中常见的软件安装方式 --------- 下载&&安装 a、yum/apt b、rpm安装包安装 c、源码安装 yum 关于 yum 的所有操作必须保证主机(虚拟机)网络畅通!!! 可以通过 ping 指令验证&#xff1a; ping www.baidu.com 安装软件 yum 会自动找到都有哪些软件包需要下载…

Ubuntu系统入门指南:常用命令详解

Ubuntu系统入门指南&#xff1a;常用命令详解 引言 Ubuntu是一个基于Linux内核的开源操作系统&#xff0c;由Canonical公司和社区共同开发和维护。它以易用性、稳定性和广泛的软件支持而著称&#xff0c;广泛应用于个人电脑、服务器和云计算环境。对于新手来说&#xff0c;掌…

Python | Leetcode Python题解之第404题左叶子之和

题目&#xff1a; 题解&#xff1a; class Solution:def sumOfLeftLeaves(self, root: TreeNode) -> int:if not root:return 0isLeafNode lambda node: not node.left and not node.rightq collections.deque([root])ans 0while q:node q.popleft()if node.left:if is…

算法-两数相加(150)

我们首先创建一个虚拟头节点dummy&#xff0c;它的主要作用是简化边界条件的处理。然后&#xff0c;我们使用一个循环来遍历两个链表&#xff0c;同时考虑进位。在循环中&#xff0c;我们计算当前位的和&#xff08;包括从上一个计算中可能遗留下来的进位&#xff09;&#xff…

南大通用数据库-Gbase-8a-学习-45-SQL优化之视图展开

目录 一、环境信息 二、参数介绍 三、实验 1、登录集群 2、测试表结构 3、测试数据 4、创建视图 5、测试SQL 6、加HINT 7、SQL效率对比 &#xff08;1&#xff09;不加HINT &#xff08;3&#xff09;加HINT 一、环境信息 名称值CPUIntel(R) Core(TM) i5-1035G1 CP…

torch.embedding 报错 IndexError: index out of range in self

文章目录 1. 报错2. 原因3. 解决方法 1. 报错 torch.embedding 报错&#xff1a; IndexError: index out of range in self2. 原因 首先看下正常情况&#xff1a; import torch import torch.nn.functional as Finputs torch.tensor([[1, 2, 4, 5], [4, 3, 2, 9]]) embedd…

Navicat使用 笔记04

Navicat调用数据库 1.创建一个自己的链接&#xff08;文件-->新建连接-->MySQL&#xff09; 进入到这个界面中&#xff1a; 【注意&#xff1a;密码是下载登录软件时设定过的】 创建一个连接完成&#xff08;通过双击激活&#xff09;。 2.在创建好的连接中创建数据库…

观后感:《中国数据库前世今生》——时代变迁中的数据库崛起

最近观看了《中国数据库前世今生》纪录片&#xff0c;这部影片详细梳理了从1980年代至今&#xff0c;中国数据库技术发展的跌宕历程。作为一名程序员&#xff0c;这部纪录片让我不禁感慨数据库技术的飞速进步&#xff0c;也让我更深入地理解了数据库技术在我们日常生活中的重要…

el-table使用el-switch选择器没效果

出现问题的代码: 0表示启用&#xff0c;1表示禁用&#xff0c;发现页面根本没有效果&#xff0c;百思不得其解&#xff0c;查阅资料&#xff0c;恍然大悟。 <el-table :data"userList" stripe border style"width: 100%" height"500"><…

AWS 将 OpenSearch 纳入 Linux 基金会旗下

AWS 今天宣布&#xff0c;随着OpenSearch 基金会的成立&#xff0c;它将把OpenSearch&#xff08;流行的 Elasticsearch 搜索和分析引擎的开源分叉&#xff09;移交给 Linux 基金会。在 Elastic 将其 Elasticsearch 和 Kibana 项目的许可证更改为自己的专有许可证 Elastic Lice…

设置PDF打开密码

为PDF文件设置打开密码是一种有效的保护措施&#xff0c;它能防止未经授权的用户访问文件内容。以下是一份专业指南&#xff0c;详细介绍如何为PDF文件设置打开密码。 打开pdf编辑器&#xff0c;我们点击工具栏中的【文件】功能&#xff0c;选择里面的【属性】 然后在属性设置…

对 Python 中 GIL 的理解

一.Python 中的 GIL Python 中的全局解释器锁&#xff08;Global Interpreter Lock&#xff0c;GIL&#xff09;是 CPython 解释器的一个机制&#xff0c;用来确保在多线程环境下&#xff0c;只有一个线程可以执行 Python 字节码&#xff0c;任何时刻只能有一个线程在执行 Pyt…

TopoDOT2024.1注册机 道路自动化提取 雷达点云数据

TopoDOT2024.1是一套成熟的点云数据处理及应用系统&#xff0c;全面具备点云数据的存储管理、精度检核、特征自动提取、智能分析、高效建模、成果输出等应用功能。TopoDOT在LiDAR数据应用领域有着多年的实战经验&#xff0c;用户在实际项目中长期使用&#xff0c;尤其在交通领域…

如何利用 Visual Studio 和 AI 工具实现高效编程

哪个编程工具让你的工作效率翻倍? 在现代软件开发的世界中,编程效率的提升对开发者来说至关重要。高效编程不仅仅是编写更多的代码,还包括如何减少重复劳动、提高代码质量、加快调试和测试流程等。而 Visual Studio 作为一个功能强大的开发环境(IDE),配合各种 AI 工具,…

JAVA-期末成绩计算

要求 总评成绩 期末成绩*0.6 平时成绩*0.4 输入总评成绩和平时成绩&#xff0c;输出期末成绩要考几分&#xff08;0<平时成绩<40&#xff0c;0<总评成绩<100&#xff09; 要求能多次运行&#xff08;退出程序停止&#xff09; 代码 import java.util.Scanne…

灾备技术演进之路 | 虚拟化无代理备份只能挂载验证和容灾吗?只能无代理恢复吗?且看科力锐升级方案

灾备技术演进之路系列 虚拟化备份技术演进 摆脱束缚&#xff0c;加速前行 无代理备份仅能挂载/恢复验证吗&#xff1f; ——科力锐极简验证演练无代理备份来了 无代理备份无法应对平台级故障吗&#xff1f; ——科力锐应急接管无代理备份来了 无代理备份仅能同平台挂载吗&a…

AWS 实时数据流服务 Kinesis

AWS 实时数据流服务 Kinesis 什么是 KinesisKinesis 数据来源示例 AWS Lambda 发送数据到 Kinesis步骤 1&#xff1a;创建 Kinesis 数据流步骤 2&#xff1a;编写 Lambda 函数步骤 3&#xff1a;配置 Lambda 函数权限部署和测试 Lambda 函数 消费和处理 Kinesis 数据流示例 Fli…

英飞凌 PSoC6 RT-Thread 评估板简介

概述 2023年&#xff0c;英飞凌&#xff08;Infineon&#xff09;联合 RT-Thread 发布了一款 PSoC™ 62 with CAPSENSE™ evaluation kit 开发板 &#xff08;以下简称 PSoC 6 RTT 开发板&#xff09;&#xff0c;该开发套件默认内置 RT-Thread 物联网操作系统。PSoC 6 RTT 开…

会计稳健性Cscore模型(2000-2022年)

参考文献 甄红线, 王三法, 王晓洪. (2019). 公司债特殊条款、债券评级与会计稳健性. 会计研究, (10), 42-49.王晓亮, 蒋勇, 刘振杰. (2019). 董事会断裂带、会计稳健性与真实盈余管理. 审计研究, (05), 120-128. 会计稳健性&#xff0c;也称为会计保守性&#xff0c;是指企业…