当前位置: 首页 > news >正文

Effective C++ 条款31:将文件间的编译依存关系降至最低

文章目录

    • 条款31:将文件间的编译依存关系降至最低
      • 最小化编译依赖关系的最佳实践
      • 通过减少编译依赖的好处
      • 总结

条款31:将文件间的编译依存关系降至最低

为了减少编译依赖关系,应该将接口与实现分离,并尽量减少头文件之间的依赖。这有助于加快编译速度,降低因代码变化引起的编译重新构建的时间。

最小化编译依赖关系的最佳实践

  1. 分离接口和实现
    将类的接口和实现分开,通常做法是将接口声明放在头文件中,而将实现代码放在源文件中。这样做可以避免头文件频繁改变导致的重新编译。
// Widget.h: 声明接口
class Widget {
public:void display();
};// Widget.cpp: 实现
#include "Widget.h"
void Widget::display() {// 实现代码
}
  1. 使用 object reference 或 object pointer 替代 object
    当使用对象时,如果对象引用或指针足以完成任务,尽量不要直接传递整个对象。通过引用或指针传递可以减少对象的拷贝,降低编译时依赖。
// 如果只需要访问对象,可以使用引用
void processWidget(const Widget& widget);  // 只传递引用
  1. 使用 class 声明式替换 class 定义式
    为了减少编译依赖,应该尽量在头文件中只提供类声明,而将类的实现放在源文件中。通过这种方式,其他文件仅依赖于声明,减少了编译时的依赖性。
// Widget.h: 仅声明
class Widget;// Widget.cpp: 定义
#include "Widget.h"
void Widget::display() {// 实现代码
}
  1. 使用 Handle class 或 Interface class
    使用 “Handle class” 和 “Interface class” 是一种常见的降低编译依赖的技术。Handle class 通常是一个不包含数据的类,只包含指针或者引用,避免了类的完整定义暴露。Interface class 是一个纯虚类,表示类的接口,所有实现细节都隐藏在派生类中。
// WidgetHandle.h: 声明类的接口
class WidgetHandle {
public:virtual void display() = 0;  // 纯虚函数
};// WidgetHandle.cpp: 实现
#include "WidgetHandle.h"
// 实现函数
  1. 使用声明式和定义式的不同头文件
    通过将声明和定义分开到不同的头文件中,可以使头文件的编译依赖最小化。声明文件只包含类的接口,而定义文件包含类的实现。
// WidgetDecl.h: 声明
class Widget;// WidgetDef.h: 定义
#include "WidgetDecl.h"
class Widget {
public:void display() { ... }
};

通过减少编译依赖的好处

  • 加快编译速度:减少依赖关系的传递,避免无关代码的重新编译,减少编译时间。
  • 提高可维护性:代码修改时不必修改过多的依赖文件,降低了维护成本。
  • 提高灵活性:依赖于声明而不是定义使得系统的扩展性和灵活性更高。修改实现不需要重新编译所有依赖该实现的文件。

总结

  • 分离接口和实现:通过分离接口和实现,减少了文件之间的依赖关系。
  • 引用或指针传递:避免直接传递对象,尽量使用引用或指针来减少拷贝和依赖。
  • 声明式替代定义式:尽可能在头文件中只提供声明,避免暴露实现细节。
  • 使用 Handle 和 Interface 类:通过 handle 类和接口类设计模式来最小化依赖关系,增强系统的灵活性。

减少编译依赖关系是提升大型项目可维护性和编译效率的关键,遵循这些原则有助于构建高效且易于扩展的系统。

http://www.lryc.cn/news/511100.html

相关文章:

  • python数据分析之爬虫基础:scrapy详解
  • openwrt 负载均衡方法 openwrt负载均衡本地源接口
  • Linux高级--3.3.2.6高并发编程之“内存屏障”“CPU屏障”“编译屏障”
  • 【含开题报告+文档+PPT+源码】基于SpringBoot的智能安全与急救知识科普系统设计与实现
  • EMQX5.X版本性能配置调优参数
  • 电脑配置maven-3.6.1版本
  • 水电站视频智能监控系统方案设计与技术应用方案
  • React 组件通信完整指南 以及 自定义事件发布订阅系统
  • 华为 AI Agent:企业内部管理的智能变革引擎(11/30)
  • 【Pandas】pandas Series empty
  • Git如何设置和修改当前分支跟踪的上游分支
  • GitHub新手用法详解【适合新手入门-建议收藏!!!】
  • 游戏开发线性空间下PS工作流程
  • 7-10 最长公共子序列
  • 亚远景-ISO 21434标准下的汽车网络安全:风险评估与管理的关键实践
  • C++ 的 source_location
  • [python SQLAlchemy数据库操作入门]-14.实时数据采集 记录股市动态
  • `we_chat_union_id IS NOT NULL` 和 `we_chat_union_id != ‘‘` 这两个条件之间的区别
  • 【和春笋一起学C++】文本输入与读取
  • D类音频应用EMI管理
  • 第N8周:使用Word2vec实现文本分类
  • 100天精通Python(爬虫篇)——第113天:爬虫基础模块之urllib详细教程大全
  • 光谱相机与普通相机的区别
  • Mysql数据 新增、修改和删除操作时,这些变化如何被转换为Kafka消息?
  • 《Python 机器视觉:开启智能视觉新时代》
  • uniapp实现为微信小程序扫一扫的功能
  • 【微信小程序】4plus|搜索框-历史搜索 | 我的咖啡店-综合实训
  • 使用FFmpeg进行拉流和推流操作
  • Unity微信小游戏接入开放数据域
  • Spring Boot的开发工具(DevTools)模块中的热更新特性导致的问题