目标

即我们最终要生成的文件,make默认生成第一个目标,注意 makefile中tab和空格不是一回事,规则使用tab缩进,编辑器不要设置诸如“将tab替换为空格之类的选项”,目标的结构如下

目标:依赖1 依赖2
<TAB>生成目标的规则

伪目标

无论如何也要执行的目标,需要使用.PHONY:标识

.PHONY:clean
clean:
rm -f *.o

常见伪目标

all 是所有目标的伪目标,功能是编译所有目标

clean 删除所有被make创建的文件

install 安装已经编译好的程序,其实就是把目标执行文件拷贝到指定目标中

print 这个伪目标的功能是列出改变过的源文件

tar 把源程序打包备份,就是一个tar文件

dist 创建一个压缩文件,一般吧tar文件压缩成Z文件或gz文件

TAGS 更新所有目标,以备完整的重编译使用

check/test 测试makefile流程

例子

circle:main.o circle.o
cc main.o circle.o -o circle
main.o:main.c
cc -c main.c
circle.o :circle.c
cc -c circle.c
clean:
rm *.o *.h circle

自动依赖头文件

makefile是根据依赖关系,时间戳和生成规则来判断哪些文件需要更新,但是我们通常写的生成规则并不会包含头文件,依赖关系也不会,这样如果我们更新了头文件,makefile也不会发现,也就不会更新相应的文件,同时,考虑到头文件包含关系的复杂性,我们在写依赖关系之前把所有的头文件理清楚再写进去也不现实,一个好的方式是利用$gcc -M$gcc -MM功能,这两个功能能自动的分析头文件的依赖关系,前者分析目标文件的所有头文件依赖关系,后者分析和我们自定义文件的依赖关系:

$gcc -M hello.c
hello.o: hello.c /usr/include/stdio.h /usr/include/features.h \
/usr/include/i386-linux-gnu/bits/predefs.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.6/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.6/include/stdarg.h \
/usr/include/i386-linux-gnu/bits/stdio_lim.h \
/usr/include/i386-linux-gnu/bits/sys_errlist.h hello.h
$gcc -MM hello.c
hello.o: hello.c hello.h

可以看到,他们的输出的格式刚好就是我们写的依赖关系的格式,所以我们可以综合利用伪目标,重定向,include关键字实现对头文件更新的自动检查

.PHONY:all
all:hello dependency
hello:hello.o
gcc hello.o -o hello
hello.o:hello.c
gcc -c hello.c -o hello.o
dependency:headers
headers:
gcc -M hello.c > headers
include headers .PHONY:clean clean:
rm -rf hello *.o

分析:make缺省目标会自动以第一个目标为目标,首先依次检查hello<-hello.o<-hello.c发现没有文件被更新,dependency并不是一个文件,也没有时间戳,所以也没有被更新,但是dependency每次都会执行刷新headers文件,我们又include了headers,所以一旦headers里面写的头文件有更新,就会通过hello.o<-xxx.h来找到更新关系,进而更新hello.o并进一步更新hello。至此实现了对头文件的自动依赖问题的解决。

执行效果:

//程序就是输出这个N值
#ifndef __HELLO_H__
#define __HELLO_H__ #define N 121220; #endif
$./hello
num:121220
$vi hello.h #修改头文件中N的值为5
$make #果然重新编译了
gcc -c hello.c -o hello.o
gcc hello.o -o hello
gcc -M hello.c > headers
$./hello
num:5

Makefile目标,伪目标,头文件自动依赖的更多相关文章

  1. makefile 自动处理头文件的依赖关系 (zz)

    现在我们的Makefile写成这样: all: main main: main.o stack.o maze.ogcc $^ -o $@ main.o: main.h stack.h maze.hst ...

  2. Makefile的伪目标

    1.Makefile伪目标的格式: .PHONY : cleanclean: rm xxxx 2.Makefile伪目标的作用: 第一种情况: 如果我们需要书写这样的一个规则:规则所定义的命令不是去创 ...

  3. 第三篇 makefile的伪目标

    我们来思考一下makefile中的目标究竟是什么?实际上,在默认情况下:    1.make将makefile的目标认为是一个文件:    2.make解释器比较目标文件和依赖文件的新旧关系,决定是否 ...

  4. makefile中伪目标的理解

    1. 我们知道Makefile中的语法是这样: target ... : prerequisites ... command - - 2. 假如编译两个文件可以这么写: a.o:a.c  gcc -c ...

  5. makefile之伪目标

    伪目标 1. 伪目标的语法: 在书写伪目标时,首先需要声明伪目标,然后再定义伪目标规则. 1.1 声明伪目标: .PHONY clean (这里声明clean是伪目标) 1.2 定义伪目标规则: cl ...

  6. Makefile中头文件在依赖关系中作用

    摘于:http://bbs.csdn.net/topics/120024677 (1)在makefile的依赖关系中用不用体现.h头文件?(2)如果在依赖关系中要体现.h头文件,应该体现到什么层次?= ...

  7. CocoaPods 导入第三方库头文件自动补齐

    使用了一段时间CocoaPods来管理Objective-c的类库,方便了不少.但是有一个小问题,当我在xcode输入import关键字的时候,没有自动联想补齐代码的功能,需要手工敲全了文件名,难以适 ...

  8. CocoaPods导入第三方库头文件自动补齐

    使用了一段时间CocoaPods来管理Objective-c的类库,方便了不少.但是有一个小问题,当我在xcode输入import关键字的时候,没有自动联想补齐代码的功能,需要手工敲全了文件名,难以适 ...

  9. jni头文件自动生成

    写一个测试代码,大概如下 package aa.bb; public class TestClass { public native void aaaBBB(); } 在项目的src目录下执行 jav ...

随机推荐

  1. Java基础复习笔记系列 九 网络编程

    Java基础复习笔记系列之 网络编程 学习资料参考: 1.http://www.icoolxue.com/ 2. 1.网络编程的基础概念. TCP/IP协议:Socket编程:IP地址. 中国和美国之 ...

  2. python 实时遍历日志文件

    首先尝试使用 python open 遍历一个大日志文件, 使用 readlines() 还是 readline() ? 总体上 readlines() 不慢于python 一次次调用 readlin ...

  3. java顺序表和树的实现

    一.顺序表 1.线性表 //java顺序表的实现,如ArrayList就是用线性表实现的,优点是查找快,缺点是添加或删除要移动很多元素,速度慢 public class SequenceList { ...

  4. 【web前端面试题整理04】阿里一行之大神面对面

    前言 这段时间我在河南一家公司当了一段时间的前端主管,最后可耻的匿了,原因各种各样,最主要的就是不想呆在郑州了. 其实这里的同事还是很不错的,面对老总最后的挽留我不是没有动心,而是这个地方确实不太好, ...

  5. 【javascript激增的思考03】MVVM与Knockout

    前言 今天搞的有点快,因为上午简单研究了下MVC,发现MVC不太适合前端开发,然后之前看几位前端前辈都推荐前端使用MVVM,但是我对其还不甚了解,所以我觉得下午还是应该先看看他是神马先,后面再决定要不 ...

  6. JavaScript强化教程 -- cocosjs场景切换

    场景切换 在main.js,将StartScene作为我们初始化运行的场景,代码如下: cc.LoaderScene.preload(g_resources, function () { cc.dir ...

  7. 关于web软件信息安全问题防护资料的整理(三)

    了解了web系统的安全威胁,那么我们应该怎样防范这些安全威胁呢? 1.时刻准备应战 Web应用系统所面临的威胁是非常严峻的.不管攻击的一方是采用单一形式的攻击,还是采用混合多种手段的混合攻击,作为防护 ...

  8. [android]AndroidInject框架——我的第一个android小型框架

    作为一个移动应用开发者,随着需求的日益增多,Android项目的越来越臃肿,代码量越来越大, 现在冷静下来回头看看我们的代码,有多少代码跟业务逻辑没什么关系的 所以,本人自不量力,在github上建了 ...

  9. Android SharedPreference的使用

    在<Android 在内部存储读写文件>一文中,谈到了登录用户名和密码的方法,通过读取内存文件来实现,但是会出现问题,因为登录名和密码的形式通过username##password的形式, ...

  10. Java你可能不知道的事系列(1)

    概述 本类文章会不段更新分析学习到的经典面试题目,在此记录下来便于自己理解.如果有不对的地方还请各位观众拍砖. 今天主要分享一下常用的字符串的几个题目,相信学习java的小伙伴们对String类是再熟 ...