Makefile规则

target ... :prerequisites...
command target就是一个目标文件,可以是object file,也可以是可以执行文件,也可以是一个标签 prerequisites就是要生成那个target所需要的文件或者目标文件 command就是make执行的命令的,任意的shell命令
target依赖于prerequisites,其生成规则定义在command中。
prerequisites中如果有一个规则以上的文件比target文件要新的话,command锁定义的命令就会被执行。

例子1

[root@typhoeus79 makefile]# ll
total 16
-rw-r--r-- 1 root root 69 Jun 20 11:27 head.c
-rw-r--r-- 1 root root 45 Jun 20 11:27 head.h
-rw-r--r-- 1 root root 150 Jun 20 11:30 Makefile
-rw-r--r-- 1 root root 81 Jun 20 11:28 test.c
------------------------------------------------------------------------
[root@typhoeus79 makefile]# more head.h
#include <stdio.h> void printword(char *s);
[root@typhoeus79 makefile]# more head.c
#include "head.h" void printword(char *s)
{
printf("%s\n",s);
}
[root@typhoeus79 makefile]# more test.c
#include "head.h" int main()
{
char *s="Hello world!"; printword(s);
}
[root@typhoeus79 makefile]# more Makefile
test: test.o head.o
gcc -o test test.o head.o head.o: head.c head.h
gcc -c head.c
test.o: test.c
gcc -c test.c clean:
rm -rf test.o head.o test

  

make是如何工作的

在默认的情况下,只输入make命令,那么:

1、make会找在当前目录下名字为makefile或者Makefile文件;

2、如果找到,它会找第一个目标文件,例如上面例子中的test文件,并把这个作为最终的目标文件

3、如果test目标文件不存在,或者edit所依赖的后面的.o文件的文件修改时间要比test文件新,那么就会执行后面所定义的命令生成test这个文件

4、如果test所依赖的.o文件也不存在,那么make会在当前文件中找目标文件为.o的文件依赖性,如果找到再根据类似3生成.o文件

整个make的依赖性,make会一层一层地区找文件的依赖关系,直到最终编译出第一个目标文件。

makefile中使用变量

PROJECT_NAME = "sqlparser"
VERSION = "2.3.0"
LDFLAGS = -lz -lm -lpthread -lcrypt -lcrypto
CFLAGS = -fPIC -Wall -W -pipe -Wno-unused-parameter -g -Wswitch -Wpointer-arith -Wredundant-decls -Wformat -D_GNU_SOURCE
OBJS = sql_parse.o sql_lex.o \
sql_select.o sql_insert.o sql_update.o sql_delete.o sql_replace.o sql_set.o sql_ddl.o sql_create_table.o sql_alter_table.o \
sql_partition.o sql_define.o sql_show.o sql_util.o sql_transaction.o sql_dml.o sql_prepared.o
TARGET = sample libsqlparse.a
LIBDIR = ./output/
CC = gcc

声明一个变量objects,通过$(objects)的方式来使用这个变量,改良版的makefile如下:

[root@typhoeus79 makefile]# more Makefile
OBJS = test.o head.o
CC = gcc test: $(OBJS)
$(CC) -o test test.o head.o head.o: head.c head.h
gcc -c head.c
test.o: test.c
gcc -c test.c clean:
rm -rf test.o head.o test
[root@typhoeus79 makefile]# make clean
rm -rf test.o head.o test
[root@typhoeus79 makefile]# make
gcc -c test.c
gcc -c head.c
gcc -o test test.o head.o

 make自动推导

make可以自动推动文件以及文件依赖关系后面的命令

只要make看到一个[.o]文件,会自动把[.c]文件加到依赖关系,并且对于的cc -c [.c]也会被推动出来

对应上面的makefile继续被修改为

[root@typhoeus79 makefile]# make clean
rm -rf test.o head.o test
[root@typhoeus79 makefile]# make
gcc -c -o test.o test.c
gcc -c -o head.o head.c
gcc -o test test.o head.o
[root@typhoeus79 makefile]# more Makefile
OBJS = test.o head.o
CC = gcc test: $(OBJS)
$(CC) -o test test.o head.o head.o: head.h clean:
rm -rf test.o head.o test

隐晦规则

.PHONY: clean
clean:
rm -rf test.o head.o test

.PHONY表示clean是个伪目标文件
另类风格的makefile

make可以自动推导命令,看到那堆[.o]和[.h]的依赖不爽,重复的[.h]是否可以收拢起来?

[root@typhoeus79 makefile]# more Makefile
OBJS = test.o printInt.o printWord.o
CC = gcc test: $(OBJS)
$(CC) -o test $(OBJS) $(OBJS): printWord.h
printWord.o:printInt.h .PHONY: clean
clean:
rm -rf $(OBJS) test
[root@typhoeus79 makefile]# make
gcc -c -o test.o test.c
gcc -c -o printInt.o printInt.c
gcc -c -o printWord.o printWord.c
gcc -o test test.o printInt.o printWord.o

这种风格,使得makefile简单,但是文件依赖关系显得凌乱,依赖关系看不清楚,对于文件多,不合适。

[root@typhoeus79 makefile]# more Makefile
OBJS = test.o printInt.o printWord.o
CC = gcc test: $(OBJS)
$(CC) -o test $(OBJS) test.o:printWord.h printWord.o:printInt.h .PHONY: clean
clean:
rm -rf $(OBJS) test
[root@typhoeus79 makefile]# make
gcc -c -o test.o test.c
gcc -c -o printInt.o printInt.c
gcc -c -o printWord.o printWord.c
gcc -o test test.o printInt.o printWord.o

 清空目标文件的规则

每个Makefile中都应该写一个清空目标文件(.o和执行文件)的规则,不仅便于重编译,也很利于保持文件的清洁。

.PHONY: clean
clean:
rm -rf $(OBJS) test

更稳健的方式,.PHONY意思clean是一个伪目标,在rm命令前面加一个小减号的意思,也许某些文件出现问题,但不要管,继续做后面的事情。

clean规则不要放在文件的开头,不然,就会变成make的默认目标。

不成文的规矩,clean从来都是放在文件的最后。

错误的例子:

.PHNOY: clean
clean:
rm -rf printInt.o

clean是目标对象,一定不能加[Tab]键,否则出错:

[root@typhoeus79 makefile]# make -f testMakefile clean
make: *** No rule to make target `clean'. Stop.

Makefile例子引入的更多相关文章

  1. Spring框架系列(2) - Spring简单例子引入Spring要点

    上文中我们简单介绍了Spring和Spring Framework的组件,那么这些Spring Framework组件是如何配合工作的呢?本文主要承接上文,向你展示Spring Framework组件 ...

  2. Makefile的引入及规则

    ARM裸机1期加强版视频课程配套WiKi第9课第5节_Makefile的引入及规则. 文字不能完全替代视频,所以如果你看了这些文章不太懂,建议购买视频进一步学习. 视频购买地址:100ask.taob ...

  3. makefile例子《一》

    一.例子 (1)makefile和src源文件不在同一目录下 (2)把.o生成到指定目录下 文件结构目录 ----inc      //放头文件 ----lib //放所需要的.a或者.so文件 -- ...

  4. 一个通用的两级Makefile例子

    目的 进行如项目的顶层目录后,运行make,即可直接编译项目中所有的源文件,并生成最终的可执行文件 实现头文件自动依赖 添加源文件不用修改Makefile,且可以自动编译新文件 顶层目录下添加文件夹, ...

  5. Linux下GCC和Makefile实例(从GCC的编译到Makefile的引入)

    一.确认已经装好了GCC和Make的软件包 可以使用whereis命令查看: 如果whereis  gcc和whereis  make命令有结果,说明安装了这两个软件,可以继续往下做. 二.使用GCC ...

  6. AngularJS2.0 hello world例子——引入这么多额外的依赖库真是很忧伤啊

    初识Angular2 写一个Angular2的Hello World应用相当简单,分三步走: 1. 引入Angular2预定义类型 import {Component,View,bootstrap} ...

  7. Linux下GCC和Makefile实例(从GCC的编译到Makefile的引入) 转

    http://www.crazyant.net/2011/10/29/linux%E4%B8%8Bgcc%E5%92%8Cmakefile%E5%AE%9E%E4%BE%8B%EF%BC%88%E4% ...

  8. (二十一)Makefile例子

    ROOT_PROJECT = .DIR_INC = -I$(ROOT_PROJECT)/include -I$(ROOT_PROJECT)/include/NE10 DIR_BIN = $(ROOT_ ...

  9. [编译] 1、第一个makefile简单例子

    前言 本篇用一个最简单的例子引入makefile,教你编写第一个makefile 正文 在Download/aa文件夹下有a.c和makefile文件 litao@litao:~/Downloads/ ...

随机推荐

  1. Dapper数据库相关操作

    using System; using System.Data; using System.Configuration; using System.Data.SqlClient; namespace ...

  2. Eclipse自动补全增强

    在Eclipse中,从Window -> preferences -> Java -> Editor -> Content assist -> Auto-Activati ...

  3. 利用python生成交换机的VRF配置文件

    为了快速生成有规律的VRF,写了一个python脚本,可以快速生成如下的VRF配置. ip vpn-instance  vpn0ipv4-family  route-distinguisher 600 ...

  4. DevOps之存储和数据库

    唠叨话 关于德语噢屁事的知识点,仅提供专业性的精华汇总,具体知识点细节,参考教程网址,如需帮助,请留言. <数据(Data)> 了解有关数据部分.涉及存储及数据库的概念:知识与技能的层次( ...

  5. Java限流策略

    概要 在大数据量高并发访问时,经常会出现服务或接口面对暴涨的请求而不可用的情况,甚至引发连锁反映导致整个系统崩溃.此时你需要使用的技术手段之一就是限流,当请求达到一定的并发数或速率,就进行等待.排队. ...

  6. Tomcat Java.OutOfMemoryError : PermGen Space异常

    背景:前些日子更新公司多年前一个旧平台发布到Tomcat上之后,频繁收到网站许多模块无法正常使用的反汇. 测试过程中发现平台发布一段时间后,访问相关网页出现如下500页面 解决方案:PermGen s ...

  7. Fedora 下 Google-Chrome 经常出现僵尸进程的权宜办法

    对于Chrome_ProcessL 和Chrome_FileThre这两僵尸进程,估计遇到过的人都对其各种无奈吧,放任不管吧,越来越多,然后卡死,只能另开个X环境或者在其他的TTY里干掉他俩再切回去, ...

  8. iOS开发从申请账号到上线APP Store步骤

    1.developer.apple.com 申请开发者账号 2.根据API Cloud创建证书: http://docs.apicloud.com/Dev-Guide/iOS-License-Appl ...

  9. JavaScript的function参数的解释

    在js里面写function时其参数在内部表示为一个数组.也就是说:我们定义一个function,里面的参数和将来调用这个function时传入的实参是毫无关系的,如果我们要定义一个function ...

  10. 2017年11月Dyn365/CRM用户社区活动报名

    UG是全球最大Dynamics的用户组织,由最终用户自发组织,由行业有经验的专家自愿贡献知识和经验的非营利机构,与会人员本着务实中立的态度,不进行推介产品,服务以及其他营销行为.在美国,微软Dynam ...