C++的主要优点是高性能, 但是说起编译速度,却只有低调的份了。如果不加以重视,编译速度极有可能会成为开发过程中的一个瓶颈。那么,为什么C++它就编译的这么慢呢?

我想最重要的一个原因应该是C++基本的"头文件-源文件"的编译模型:

  1. 每个源文件作为一个编译单元,可能会包含上百甚至上千个头文件,而在每一个编译单元,这些头文件都会被从硬盘读进来一遍,然后被解析一遍。
  2. 每个编译单元都会产生一个obj文件,然后所以这些obj文件会被link到一起,并且这个过程很难并行。

这里,问题在于无数头文件的重复load与解析,以及密集的磁盘操作。

下面从各个角度给出一些加快编译速度的做法,主要还是针对上面提出的这个关键问题。

1. 代码角度

1.1. 在头文件中使用前置声明,而不是直接包含头文件

不要以为你只是多加了一个头文件,由于头文件的"被包含"特性,这种效果可能会被无限放大。所以,要尽一切可能使头文件精简。很多时候前置申明某个namespace中的类会比较痛苦,而直接include会方便很多,千万要抵制住这种诱惑;类的成员,函数参数等也尽量用引用,指针,为前置声明创造条件。

1.2. 使用Pimpl模式

Pimpl全称为Private Implementation。传统的C++的类的接口与实现是混淆在一起的,而Pimpl这种做法使得类的接口与实现得以完全分离。如此,只要类的公共接口保持不变,对类实现的修改始终只需编译该cpp;同时,该类提供给外界的头文件也会精简许多。

1.3. 高度模块化

模块化就是低耦合,就是尽可能的减少相互依赖。这里其实有两个层面的意思。一是文件与文件之间,一个头文件的变化,尽量不要引起其他文件的重新编译;二是工程与工程之间,对一个工程的修改,尽量不要引起太多其他工程的编译。这就要求头文件,或者工程的内容一定要单一,不要什么东西都往里面塞,从而引起不必要的依赖。这也可以说是内聚性吧。

以头文件为例,不要把两个不相关的类,或者没什么联系的宏定义放到一个头文件里。内容要尽量单一,从而不会使包含他们的文件包含了不需要的内容。记得我们曾经做过这么一个事,把代码中最"hot"的那些头文件找出来,然后分成多个独立的小文件,效果相当可观。

其实我们去年做过的refactoring,把众多DLL分离成UI与Core两个部分,也是有着相同的效果的 - 提高开发效率。

1.4. 删除冗余的头文件

一些代码经过上十年的开发与维护,经手的人无数,很有可能出现包含了没用的头文件,或重复包含的现象,去掉这些冗余的include是相当必要的。当然,这主要是针对cpp的,因为对于一个头文件,其中的某个include是否冗余很难界定,得看是否在最终的编译单元中用到了,而这样又可能出现在一个编译单元用到了,而在另外一个编译单元中没用到的情况。
之前曾写过一个Perl脚本用来自动去除这些冗余的头文件,在某个工程中竟然去掉多达了5000多个的include。

1.5. 特别注意inline和template

这是C++中两种比较"先进"的机制,但是它们却又强制我们在头文件中包含实现,这对增加头文件的内容,从而减慢编译速度有着很大的贡献。使用之前,权衡一下。

2. 综合技巧

2.1. 预编译头文件(PCH)

把一些常用但不常改动的头文件放在预编译头文件中。这样,至少在单个工程中你不需要在每个编译单元里一遍又一遍的load与解析同一个头文件了。

2.2. Unity Build

Unity Build做法很简单,把所有的cpp包含到一个cpp中(all.cpp) ,然后只编译all.cpp。这样我们就只有一个编译单元,这意味着不需要重复load与解析同一个头文件了,同时因为只产生一个obj文件,在链接的时候也不需要那么密集的磁盘操作了,估计能有10x的提高,看看这个视频感受一下其做法与速度吧。

2.3. ccache

compiler cache, 通过cache上一次编译的结果,使rebuild在保持结果相同的情况下,极大的提高速度。我们知道如果是build,系统会对比源代码与目标代码的时间来决定是否要重新编译某个文件,这个方法其实并不完全可靠(比如从svn上拿了上个版本的代码),而ccache判断的原则则是文件的内容,相对来讲要可靠的多。很可惜的是,Visual Studio现在还不支持这个功能 - 其实完全可以加一个新的命令,比如cache build,介于build与rebuild之间,这样,rebuild就可以基本不用了。

2.4. 不要有太多的Additional Include Directories

编译器定位你include的头文件,是根据你提供的include directories进行搜索的。可以想象,如果你提供了100个包含目录,而某个头文件是在第100个目录下,定位它的过程是非常痛苦的。组织好你的包含目录,并尽量保持简洁。

3. 编译资源

要提高速度,要么减少任务,要么加派人手,前面两个方面讲得都是减少任务,而事实上,在提高编译速度这块,加派人手还是有着非常重要的作用的。

3.1. 并行编译

编译的时候指定使用多核;买个4核的,或者8核的cpu,每次一build,就是8个文件并行着编,那速度,看着都爽。

3.2. 更好的磁盘

我们知道,编译速度慢很大一部分原因是磁盘操作,那么除了尽可能的减少磁盘操作,我们还可以做的就是加快磁盘速度。比如上面8个核一块工作的时候,磁盘极有可能成为最大的瓶颈。买个15000转的磁盘,或者SSD,或者RAID0的,总之,越快越好。

3.3. 分布式编译

一台机子的性能始终是有限的,利用网络中空闲的cpu资源,以及专门用来编译的build server来帮助你编译才能从根本上解决我们编译速度的问题,想想原来要build 1个多小时工程的在2分钟内就能搞定,你就知道你一定不能没有它 - Incredibuild。

3.4. 并行,其实还可以这么做

这是一个比较极端的情况,如果你用了Incredibuild,对最终的编译速度还是不满意,怎么办?其实只要跳出思维的框架,编译速度还是可以有质的飞跃的 - 前提是你有足够多的机器:

假设你有solution A和solution B,B依赖于A,所以必须在A之后Build B。其中A,B Build各需要1个小时,那么总共要2个小时。可是B一定要在A之后build吗?跳出这个思维框架,你就有了下述方案:

  • 同时开始build A和B 。
  • A的build成功,这里虽然B的build失败了,但都只是失败在最后的link上。
  • 重新link B中的project。

这样,通过让A的build与B的编译并行,最后link一下B中的project,整个编译速度应该能够控制在1个小时15分钟之内。

另外,这本书谈了很多这方面的内容: 大规模C++程序设计。

4. 分拆文件

在开发过程中经常遇到这样的情况:在某段时间内需要只在某个文件上开发或者调试。如果该文件比较大或者依赖该文件的文件比较多,则每次哪怕只改动一行,编译时间都可能很长。

解决这种问题的方法是:把该文件分拆成多个文件,让比较稳定的部分独立成一个文件,变动比较大的部分独立成一个文件。

这种方法有时可能违背代码设计模式的原则,所以只能用于调试或者开发中的不稳定阶段。

参考文献

  • https://blog.csdn.net/weixin_34111819/article/details/85443506?utm_medium=distribute.pc_relevant.none-task-blog-OPENSEARCH-1.control&depth_1-utm_source=distribute.pc_relevant.none-task-blog-OPENSEARCH-1.control

C++加快编译速度的方法相关推荐

  1. make太慢了,加快编译速度的方法 make -j

    make太慢了,加快编译速度的方法 make -j 2018-01-18 09:04:05 gonghuihuihui 阅读数 21957  收藏 更多 分类专栏: linux 版权声明:本文为博主原 ...

  2. VS2013 加快编译速度 的方法整理

    VS2013 加快编译速度 的方法整理 1.更改项目设置 项目|属性|C/C++|代码生成|启用最小重新生成:Yes(/Gm) 项目|属性|C/C++|常规|调试信息格式:程序数据库(/Zi) 项目| ...

  3. 跑linux编译什么CPU速度快,linux 加快编译速度

    <操作系统>课程设计报告课程设计题目:操作系统课程设计 设计时间:2016/1/10一. 课程设计目的与要求需要完成的内容:(1) 安装虚拟机:Vmware.Vmware palyer ( ...

  4. 慢腾腾的Quartus prime16.0加快编译速度

    前言 当一个工程反复修改的时候,可能有时候源代码没有更改,为了加快编译速度可以配置quartus一些选项.当然,初次编译的速度是否会提升,未验证.更高级的设计分区以及逻辑锁区提升速度,以后阐述. 流程 ...

  5. win7计算机怎么优化驱动器,Win7优化电脑加快关机速度的方法技巧

    原标题:Win7优化电脑加快关机速度的方法技巧 在使用Win7系统的过程中,很多用户都遇到过关机速度慢的问题,怎么办呢?电脑的关机速度慢和开机速度慢其实是一个道理,一般是由于磁盘.电源等问题造成的.这 ...

  6. 【转】Linux程序编译速度提高方法

    项目越来越大,每次需要重新编译整个项目都是一件很浪费时间的事情.Research了一下,找到以下可以帮助提高速度的方法,总结一下. tmpfs 有人说在Windows下用了RAMDisk把一个项目编译 ...

  7. Linux程序编译速度提高方法

    2019独角兽企业重金招聘Python工程师标准>>> 项目越来越大,每次需要重新编译整个项目都是一件很浪费时间的事情.Research了一下,找到以下可以帮助提高速度的方法,总结一 ...

  8. 提高编译速度的方法——ccache的使用

    在使用CMake编译大型工程时.如果每次只是单独对一个文件进行修改还好,CMake可以针对修改单独编译,但往往只要对CMakeLists做一点小修改,整个工程就要重新编译,对于普通性能的电脑来说通常非 ...

  9. 【修改注册表加快上网速度的方法】

    如今上网已经成为大家必不可少的东西了,如果网络速度慢,那还真是要人命,像类似这样的情况有没有办法解决呢? 如果您所用的局域网网络特别不好,或是使用宽频方式上网,那么在注册表中设定适当的TcpWindo ...

最新文章

  1. ajax跨域异常,ajax跨域问题
  2. 高并发编程-线程生产者消费者的综合示例
  3. 音视频技术开发周刊 77期
  4. P3899 [湖南集训]更为厉害(线段树合并、长链剖分、二维数点)
  5. jooq代码生成_将jOOQ与Spring结合使用:代码生成
  6. 获取上周_上周惠州13盘预售9盘价格涨了!最高涨1000元/㎡
  7. 美团技术:百亿规模API网关服务Shepherd的设计与实现
  8. CSS3 背景起始位置 background-origin属性
  9. 测试人员的系统性思维
  10. egret性能优化总结
  11. R语言空间权重矩阵columbus及画图
  12. 微信小程序:强大的多功能图片处理器
  13. 产品经理面试——简历填写
  14. Centos7模板机制作
  15. MISC互相伤害!!!
  16. Timing Constraint介绍-Design Compiler(三)
  17. HTML+CSS一个简单的电影网页制作作业,学生个人html静态网页制作成品代码, html电影速递网
  18. Word中遇到的各种问题及解决方案
  19. bootloader详解----bootloader的基本概念
  20. 2020-11-02

热门文章

  1. windows服务搭建及命令总结
  2. Content-Type: application/vnd.ms-excel
  3. Python图片处理PIL/pillow/生成验证码/出现KeyError: 和The _imagingft C module is not installed...
  4. QT开发(五十)——QT串口编程基础
  5. Android Service
  6. Citrix XenServer@cloudstack基本功能测试报告2
  7. min聚合函数查询带有额外字段sql|dense_rank()over(partition)|+班级学生成绩最高
  8. Coolite Toolkit学习笔记六:常用控件Accordion、ToolBar、ToolTip
  9. 面向JavaScript开发人员的Adobe AIR与Dreamweaver
  10. GO语言有哪些优势?怎样入门?