有main.c:

 #include <stdio.h>
#include "command.h" int main(int argc, const char *argv[])
{
printf("run in main\n");
return ;
}

commad.h:

 #ifndef __COMMAND_H__
#define __COMMAND_H__ #define PI 3.14159 #endif

  之前写的makefile都是使用下面这种形式的静态规则:

 OBJ+=main.o

 $(OBJ):%.o:%.c
$(CC) -c $(CFLAGS) $< -o $@

  这种方法有个比较致命的缺点,比如main.c如果包含了一个头文件commad.h,我改动了这个头文件,由于main.o的依赖中没包含commad.h,所以main.o不会重新编译,显然这不是我想要的结果,通常我的做法是如果我修改了commad.h里面的一个宏MY_MCRO,使用souce insight的搜索共功能

  将最后一个选项: Touch files and cause recompile勾选上,这样被搜索到的文件就被touch了,或者麻烦点自己用shell语句实现同样的功能。但是如果.h文件里面如果改动了很多地方,这种做法显然太麻烦了,如果gcc的-M或者-MM选项那么就可以完美解决这个问题。

  首先gcc -M main.c输出:

  main.o: main.c /usr/include/stdc-predef.h /usr/include/stdio.h \
     /usr/include/features.h /usr/include/i386-linux-gnu/sys/cdefs.h \
     /usr/include/i386-linux-gnu/bits/wordsize.h \
     /usr/include/i386-linux-gnu/gnu/stubs.h \
     /usr/include/i386-linux-gnu/gnu/stubs-32.h \
     /usr/lib/gcc/i686-linux-gnu/4.8/include/stddef.h \
     /usr/include/i386-linux-gnu/bits/types.h \
     /usr/include/i386-linux-gnu/bits/typesizes.h /usr/include/libio.h \
     /usr/include/_G_config.h /usr/include/wchar.h \
     /usr/lib/gcc/i686-linux-gnu/4.8/include/stdarg.h \
     /usr/include/i386-linux-gnu/bits/stdio_lim.h \
     /usr/include/i386-linux-gnu/bits/sys_errlist.h command.h
 gcc -MM main.c输出:

  main.o: main.c command.h

  这就是目标与依赖的关系,利用gcc的预编译功能,找到目标依赖的.c .h文件,这个选项是专门为makefile准备的,-MM只输出非系统的.h文件,系统头文件不会变,所以一般用-MM。一个模板:

 objects = main.o

 CFLAGS:= -g

 main:$(objects)
cc $(objects) -o main -include $(objects:.o=.d)
%.d:%.c
@set -e; rm -f $@; \
$(CC) -MM $< > $@.$$$$; \
sed 's,\($*\)\.o[ : ]*,\1.o $@ : ,g' < $@.$$$$ > $@; \
rm -f $@.$$$$ .PHONY:clean
clean:
@-rm main -f $(objects) *.d *.d.*

  整个makefile的执行过程,先include文件,发现main.d不存在,但是下面有main.d的生成规则, main.d:main.c, main.d依赖于main.c,通过下面的规则生成了main.d文件,main.d文件的内容是:

main.o main.d : main.c command.h

  因此执行完include,实际得到的makefile会是这个样子:

objects = main.o

CFLAGS:= -g

main:$(objects)
cc $(objects) -o main main.o main.d : main.c command.h %.d:%.c
@set -e; rm -f $@; \
$(CC) -MM $(INCLUDE_DIRS) $< > $@.$$$$; \
sed 's,\($*\)\.o[ : ]*,\1.o $@ : ,g' < $@.$$$$ > $@; \
rm -f $@.$$$$ .PHONY:clean
clean:
@-rm main -f $(objects) *.d *.d.*

  这样以来,如果改动了main.c文件会导致 main.d文件和main.o文件重新编译或生成,改动command.h文件也会导致main.o main.d文件重新编译或生成。

  有一点需要稍微注意,就是include和生成.d文件那段最好放在终极目标的后面,否则生成终极目标的时候需要显示指明了,比如:make main, 因为如果include 和 %.d:%.c那段如果放终极目标之前,终极目标不是第一个目标了,makefile默认执行第一个目标。

makefile自动生成目标与依赖的关系的更多相关文章

  1. Makefile自动生成头文件依赖

    前言 Makefile自动生成头文件依赖是很常用的功能,本文的目的是想尽量详细说明其中的原理和过程. Makefile模板 首先给出一个本人在小项目中常用的Makefile模板,支持自动生成头文件依赖 ...

  2. makefile自动生成依赖关系

    手工编写依赖关系不仅工作量大而且极易出现遗漏,更新也很难及时,修改源或头文件后makefile可能忘记修改.为了解决这个问题,可以用gcc的-M选项自动生成目标文件和源文件的依赖关系.-M选项会把包含 ...

  3. Makefile中自动生成头文件依赖

    为什么需要自动生成头文件依赖? 编译单个源文件时,需要获取文件中包含的头文件的信息,但是一般的Makefile不会在规则中明确写明文件依赖的头文件,所以单独修改头文件后,不会导致包含头文件的源文件重新 ...

  4. Linux Makefile自动生成--config.h

    Linux Makefile自动生成--config.h http://blog.csdn.net/spch2008/article/details/12510805

  5. Makefile 自动生成依赖

    虽然以前对Makefile有个基本概念,但是真正到自己去写一个哪怕是简单的Makefile时也会遇到不少的麻烦.    现在我有如下文件 dList.h dList.c memory.c debug. ...

  6. pip自动生成requirements.txt依赖关系清单

    Python项目中经常会带requirements.txt文件,里面是项目所依赖的包的列表,也就是依赖关系清单,这个清单也可以使用pip命令自动生成. pip命令: 1 pip freeze > ...

  7. Makefile自动生成

    automake/autoconf入门作为Linux下的程序开发人员,大家一定都遇到过Makefile,用make命令来编译自己写的程序确实是很方便.一般情况下,大家都是手工写一个简单Makefile ...

  8. makefile自动生成学习

    https://www.cnblogs.com/jrglinux/p/6964169.html 关键是如何写Makefile.am  其他的交给 自动工具完成 添加一个 很好的博客 学习下 https ...

  9. 使用pip命令自动生成项目安装依赖清单

    Python项目中经常会带requirements.txt文件,里面是项目所依赖的包的列表,也就是依赖关系清单,这个清单也可以使用pip命令自动生成. pip命令: 1 pip freeze > ...

随机推荐

  1. JQ关于浏览器宽高的获取方式

    JQ关于浏览器宽高的获取方式 alert($(window).height()); //浏览器时下窗口可视区域高度alert($(document).height()); //浏览器时下窗口文档的高度 ...

  2. 遇上了artTemplate做的东西

    js现在最牛的地方是 有了Node.js后,前后端的界限几乎都消失了,围绕着它,出现了一整套生态体系. 在生态方面,比php好太多了.

  3. 小白学Linux(二)--命令行基本操作

    安装完Ubuntu后,进入系统,呈现在眼前的是Ubuntu的界面,跟windows的差不太多.一般操作系统包含GUI和CLI.GUI就是我们现在看到的,也是windows常用的直接用拖拽,点击等操作对 ...

  4. php获取textarea的值并处理回车换行的方法

    //注:\n是用双引号包的的,双引号!!双引号!!! explode("\n",$row[0]['value']

  5. Hibernate关联映射及高级查询

    一.Hibernate中的关联关系 1.1.单向一对多关联关系 按照以下步骤配置hibernate中持久化类的一对多对象关联: (1).持久化类添加关联类的相关属性及getter/setter方法. ...

  6. window10 mysql5.7 解压版 安装

    1. 解压mysql-5.7.11-winx64.zip 到某文件夹, 如C:\DevelopCommon\mysql-5.7.11-winx64. 2. 配置环境变量 变量名 : MYSQL_HOM ...

  7. T-SQL的回车和换行符(SQL)

    T-SQL的回车和换行符(SQL) sql server中的回车换行字符是  char(13)+char(10) 回车:char(13) 换行:char(10) 实例1: DECLARE @c NVA ...

  8. jQuery淡入淡出效果轮播图

    用JavaScript做了平滑切换的焦点轮播图之后,用jQuery写了个简单的淡入淡出的轮播图,代码没有做优化,html结构稍微有一些调整,图片部分用ul替换了之前用的div. html结构如下: & ...

  9. SQL2012 提示评估已过期 解决方案- sql server问题

    SQL2012 提示评估已过期 解决方案提示评估已过期的解决方法和 sql2008一样 第1步:进入SQL2012配置工具中的安装中心第2步:再进入维护界面,选择版本升级第3步:进入产品密钥,输入密钥 ...

  10. 在SharePoint中无代码开发InfoPath应用: 一个测试Web Service的工具

    这是这个系列的第一篇,介绍一个小工具,主要是用在Web Service测试的. 因为为了用一点高级的东西,就免不了和web service打交道. 你可以使用按照KB819267来修改web.conf ...