文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Makefile-只修改了.h头文件,编译为什么不起作用?

2024-12-03 03:38

关注

不知道各位小伙伴是否碰到过这样的情况:

一个 .c 文件 include 另一个 .h 头文件,使用 Makefile 来构建(编译)应用程序。

第一次编译、执行,很正常!

但是此时,如果修改了 .h 头文件,再次编译时,就出现问题了:

预期的执行流程是:make 发现 .h 头文件的修改时间更新,于是重新编译包含这个头文件的所有 .c 文件。

可实际的结果却是:make 并没有识别出 .h 头文件的修改。

这是怎么回事呢?让我们一一道来。

简单的代码示例

一个头文件:hello.h

  1. #ifndef _HELLO_ 
  2. #define _HELLO_ 
  3.  
  4. #define  NUM    1 
  5.  
  6. #endif 

一个源文件:main.c

  1. #include  
  2. #include "hello.h" 
  3.  
  4. int main(int argc, char *agv[]) 
  5.     printf("NUM = %d \n", NUM); 
  6.     return 0; 

Makefile 文件:

  1. OBJS := main.o  
  2. TARGET := main 
  3.  
  4. all : $(OBJS)  
  5.     gcc -o $(TARGET) $(OBJS) 
  6.  
  7. %.o: %.c  
  8.     gcc $< -c -o $@ 

现在我们来第一次执行 make,编译一下:

  1. $ make 
  2. gcc main.c -c -o main.o 
  3. gcc -o main main.o 

执行一下:

  1. $ ./main  
  2. NUM = 1 

我们现在把 hello.h 文件中的 NUM 改成 2,现在的文件修改时间是:

  1. $ ll 
  2. total 28 
  3. -rw-rw-r-- 1 root root   58 Jun  7 20:52 hello.h 
  4. -rwxrwxr-x 1 root root 8608 Jun  7 20:51 main* 
  5. -rw-rw-r-- 1 root root  122 Jun  7 20:51 main.c 
  6. -rw-rw-r-- 1 root root 1528 Jun  7 20:51 main.o 
  7. -rw-rw-r-- 1 root root  100 Jun  7 20:51 Makefile 

然后再执行 make 指令,编译一下:

  1. $ make 
  2. gcc -o main main.o 

可以看到:make 只执行了 Makefile 中的链接指令(从目标文件 main.o 到可执行文件 main),并没有执行 gcc main.c -c -o main.o 这条编译指令来重新编译目标文件。

也就说明:make 并没有识别出 hello.h 这个头文件已经被改动了,尽管它“应该”可以从文件的修改时间上发现!

为什么会这样?

我们来看一下 Makefile 中的这个规则:

  1. %.o: %.c  
  2.     gcc $< -c -o $@ 

目标文件 main.o,只是依赖了 main.c 文件,并没有依赖 hello.h 文件。

make 的执行规则是:只有目标文件不存在,或者依赖文件比目标文件更新的时候,才会执行编译指令。

因此,虽然 hello.h 被修改了,但是它并不是目标文件 main.o 的依赖。

make 发现:main.o 在当前目录中是已经存在的,并且它比 main.c 更新,因此不会重新编译 main.o。

所以即使 hello.h 被修改了,也不会起作用,因为 make 压根就不把 hello.h 当做 main.o 的依赖!

注意:所有的操作过程没有执行 clean 操作。

最简单、无脑的方法

既然知道了原因,那就好办了,我们手动把头文件 hello.h 加到依赖中,不就可以了吗?!

把 Makefile 中最后面几句修改成下面这样:

  1. HEADERS := hello.h 
  2. %.o: %.c ${HEADERS} 
  3.     gcc $< -c -o $@ 

也就是把 .h 文件,也加入到 .o 文件的依赖中,这样的话,每次修改 .h 文件后,再执行 make 指令时,就可以重新编译 .o 目标文件了。

您可试一下,这样做肯定是没有问题的。

到此,问题是被解决了,但是总觉得这样的方式比较粗鲁。

想一下:如果有很多的 .c 和 .h 文件呢,总不能手动一个一个添加吧?

高级一点的方法

修改 Makefile 为下面这样:

  1. OBJS := main.o  
  2. TARGET := main 
  3.  
  4. all : $(OBJS)  
  5.     gcc -o $(TARGET) $(OBJS) 
  6.  
  7. -include *.d 
  8. %.o: %.c  
  9.     gcc $< -c -MMD -o $@ 

改动部分有 2 处:

添加了 -include *.d 指令;

gcc 编译指令中,添加了 -MMD 参数;

我们先执行一下试试。第一次编译:

  1. $ ll      // 查看当前文件 
  2. total 12 
  3. -rw-rw-r-- 1 root root  58 Jun  7 21:06 hello.h 
  4. -rw-rw-r-- 1 root root 122 Jun  7 20:51 main.c 
  5. -rw-rw-r-- 1 root root 119 Jun  7 21:05 Makefile 
  6. $  
  7. $ make    // 编译 
  8. gcc main.c -c -MMD -o main.o 
  9. gcc -o main main.o 
  10. $  
  11. $ ll      // 再次查看当前文件 
  12. total 32 
  13. -rw-rw-r-- 1 root root   58 Jun  7 21:06 hello.h 
  14. -rwxrwxr-x 1 root root 8608 Jun  7 21:06 main* 
  15. -rw-rw-r-- 1 root root  122 Jun  7 20:51 main.c 
  16. -rw-rw-r-- 1 root root   23 Jun  7 21:06 main.d 
  17. -rw-rw-r-- 1 root root 1528 Jun  7 21:06 main.o 
  18. -rw-rw-r-- 1 root root  119 Jun  7 21:05 Makefile 
  19. $  
  20. $ ./main  // 执行 
  21. NUM = 1 

有没发现:多出了一个文件 main.d,该文件内容是:

  1. main.o: main.c hello.h 

这个文件正是因为 Makefile 中的 -MMD 这个参数导致生成的,而它的内容正是我们需要的目标文件依赖信息。

然后在 Makefile 中,include 这个 .d 文件,从而让 make 知道:main.o 文件依赖于 main.c 和 hello.o 这 2 个文件。

这个时候,我们再来修改 hello.h 中的内容,例如:把 NUM 改成 10,再次编译、执行:

  1. $ make 
  2. gcc main.c -c -MMD -o main.o 
  3. gcc -o main main.o  
  4. $ ./main 
  5. NUM = 10 
本文转载自微信公众号「IOT物联网小镇」,可以通过以下二维码关注。转载本文请联系IOT物联网小镇公众号。

 

来源: IOT物联网小镇内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯