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

辽宁省工程招标网广安seo

辽宁省工程招标网,广安seo,网站网站,网站后台fpt博主参与了一个使用qmake构建的项目,包含几百个源文件,最近遇到一个恼人的问题:有时仅仅修改了一个.cpp文件,构建项目时就有可能触发全编译。但是编译时又会命中ccache的缓存,这说明源代码实际上内容并没有发生变化。即…

博主参与了一个使用qmake构建的项目,包含几百个源文件,最近遇到一个恼人的问题:有时仅仅修改了一个.cpp文件,构建项目时就有可能触发全编译。但是编译时又会命中ccache的缓存,这说明源代码实际上内容并没有发生变化。即使命中了ccache缓存,几百个源文件编译下来还是要耗一小会儿时间的,博主对此不能熟视无睹。

本文中使用了一个demo项目stupid_qmake来复现和分析该问题,其结构非常简单:

stupid_qmake/
├── main.cpp
├── stupid_qmake.pro
└── utility├── foo.cpp└── foo.h

main.cpp文件内容如下,调用了std::swap以及标准输出流:

#include <iostream>
#include <utility>int main() {int a = 1;int b = 2;std::swap(a, b);std::cout << a << " " << b << std::endl;return 0;
}

正常来说,如果我们修改了一个.h文件,那么所有依赖这个.h文件的.cpp文件都需要重新编译,无论是直接include还是间接include;而修改一个.cpp文件,则重新编译这个.cpp就足够了。

这些重新编译的触发依赖于构建系统,以下面的Makefile为例:

CXX = g++
CXXFLAGS = -Wall -g
TARGET = my_programSRCS = main.cpp func.cpp
OBJS = main.o func.oall: $(TARGET)$(TARGET): $(OBJS)$(CXX) $(CXXFLAGS) -o $@ $^main.o: main.cpp func.h$(CXX) $(CXXFLAGS) -c main.cpp -o main.ofunc.o: func.cpp func.h$(CXX) $(CXXFLAGS) -c func.cpp -o func.oclean:rm -f $(OBJS) $(TARGET).PHONY: all clean

不算伪目标,共有3个target:my_programmain.ofunc.o。target和target之间的依赖,以及target对源文件的依赖,如下图所示:

my_program
main.o
func.o
main.cpp
func.h
func.cpp

在执行make时,make不会也不可能真的去检查源文件内容是否发生了变化,而是会根据源文件和target的最后修改时间(mtime)以及target之间的依赖关系,来决定哪些target需要重新生成:如果源文件的mtime比target的mtime大,说明源文件有更新,这个target需要重新生成,同时所有依赖这个target的其他target也需要重新生成。

在我们的demo项目中,从main.cpp的内容来看,修改foo.h/foo.cpp不应当导致main.cpp重新编译,但实际情况并非如此:每当使用qtcreator编辑foo.cpp后,总会触发main.cpp的重新编译。

这个问题分析起来的入手点就是看看qmake到底为我们生成了一个怎样的Makefile —— 如果你在命令行中编译过qmake项目,你应该知道,运行qmake命令时会在构建目录中生成一个Makefile文件,然后再运行make命令才会正式开始项目的编译。

stupid_qmake.pro文件生成的Makefile中,main.ofoo.o两个target的生成规则如下:

main.o: ../main.cpp ../utility$(CXX) -c $(CXXFLAGS) $(INCPATH) -o main.o ../main.cppfoo.o: ../utility/foo.cpp ../utility/foo.h$(CXX) -c $(CXXFLAGS) $(INCPATH) -o foo.o ../utility/foo.cpp

foo.o的生成规则没什么问题,main.o的生成规则看起来有点奇怪:在依赖项中竟然有一个utility目录。它为什么会在依赖项中呢?联想到main.cpp源文件中依赖了utility头文件,我们可以猜测,qmake在生成依赖规则时,utility目录被错误地视为了utility头文件被添加到了main.cpp的依赖项中。为了验证这个猜测,我们把main.cpp中对utility头文件的依赖去掉,然后重新运行qmakemain.cpp的生成规则就变成了:

main.o: ../main.cpp $(CXX) -c $(CXXFLAGS) $(INCPATH) -o main.o ../main.cpp

没有那条对utility目录的依赖了,所以我们的推测是正确的,是qmake混淆了utility目录和utility头文件。

依赖项列表中有一项是目录会产生什么后果呢?make似乎并不介意这件事,仍然会机械地扫描依赖项的mtime,以此决定哪些target需要重新生成。所以现在需要探讨的问题是:目录的mtime在什么情况下会更新?关于这个问题,我在另外一篇博客Linux:使用vim编辑文件为什么会影响目录的mtime中分析过,在那篇博客中我是以vim为例来分析的,实际上qtcreator也有相同的效果:当你使用qtcreator编辑了某个目录下的文件后,这个目录的mtime就会更新。所以说,在我们的demo项目中,如果你编辑了utility目录下的文件,utility目录的mtime就会更新,进而引起main.cpp的重新编译。

回到博文开头提到的那个项目,里面恰好有一个utility目录,而且这个目录下的源文件的修改也比较频繁。那么现在我们可以还原出整个问题的全貌:C++标准库中的utility头文件是一个被广泛包含的头文件,整个项目中的大部分源文件都对它有直接或者间接的依赖。而qmake错误地将项目中一个名为utility的目录当成了utility头文件这件事,就会导致utility目录被添加到了大部分.o文件的依赖项中。一旦我们编辑了utility目录下的文件,utility目录的mtime就会被更新,这将会导致大部分源文件重新编译。

问题修改起来也简单:在将utility目录重命名为util后,恼人的问题就消失了。

(按理说,遇到这种情况时,qmake应当优先匹配系统目录下的头文件,有时间了研究一下qmake的源码看看它为什么没有这么做)

http://www.yayakq.cn/news/839938/

相关文章:

  • 上海工商网站官网江西网站建设技术
  • 成都旅游的网站建设宁夏手机网站建设
  • 青岛网站建设外贸网站推广方案及预算
  • 做网站有地区差异吗济南天桥区做网站公司
  • 网站怎么制作的我想投资谁有项目
  • 网站前台显示数据库指定分类怎么做php网站开发和网络工程哪个好
  • 优化网站搜索排名网约车多少钱一辆
  • 重庆求建网站互联网seo是什么意思
  • 招投标建设网站的网站亿寻跨境外贸人才网
  • 移动网站开发 书wordpress会员中心模板下载
  • 厦门企业网站公司灰色行业关键词推广
  • 网站备案名称规则青岛网站建设公司外包
  • 淘宝禁止了网站建设类温州网络科技有限公司
  • 淘宝网站建设协议网站关键字没有排名
  • 贵阳网站建设网站制作天眼查企业查询app
  • 潍坊网络优化排名公众号关键词排名优化
  • 电子商务网站建设规划论文网站添加搜索
  • 无锡网站设杭州品牌设计公司有哪些
  • 餐饮网站开发性能需求分析网站建设公司合肥
  • 网站内部优化是什么意思设计公司推广渠道
  • 专业建站公司前景做网站怎么样才能赚到钱
  • 注册网站会不会有问题网站设计与制作专业
  • 关于做电影的网站设计福州免费自助建站模板
  • 用阿里云做网站如何查看网络服务商
  • 高端大气的网站制作网站搭建网站
  • 渭南网站制作学校哈尔滨网站建设v5star
  • 深圳网站程序开发验证码平台网站开发
  • 做企业网站一定要企业邮箱嘛湖北百度seo排名
  • 建设的网站首页企业app有哪些
  • 网站建设的核心是在建工程