Make和Makefile编写(详见GCC手册)
Makefile和Make Rules
多模块软件、依赖树和Make
默认规则
Make使用程序对简单变量的支持
内建变量
虚目标
特殊目标
一般性语法错误及其纠正措施
命令行的使用和调试
Makefile中常用规则总结
make常用选项
-d 显示调试信息
-f 指定从哪个文件中读取依赖关系信息。默认文件是“Makefile”或“makefile” ,"-"表示从标准输入
-h 显示所有的Makefile的help信息
-n 打印所有Makefile执行命令,但不执行这些命令
-s 运行时不显示任何信息
注释
#
连接符
\
关联列表和命令列表中使用shell通配符
?
*
默认模式规则
%.o:%.c:
$(CC) $(CFLAGS) -c $<
%.o:%.s:
$(AS) $(ASFLAGS) -o $@ $<
简单变量
定义:变量名:=文本
添加:变量名+=文本
变量引用
$(变量名)
${变量名}
$单字符变量
C := gcc
$C
CC := gcc
OPTIONS := -O3
OBJECTS :=main.o
OBJECTS := input.o compute.o
SOURCE :=main.c input.c compute.c
HEADERS := main.h input.h compute.h
power: $(OBJECTS)
$(CC) $(OPTIONS) $(OBJECTS) -o power -lm
main.o: main.h input.h compute.h #包含隐含规则
input.o: input.h
compute.o: compute.h
tar: Makefile $(HEADERS) $(SOURCES) #伪目标tar用于打包Makefile和源文件头文件
tar -cvf power.tar Makefile $(HEADERS) $(SOURCES)
clean:
rm *.o
内置变量
$@ 当前目标的名称
$? 比当前目标更加新的已修改的依赖性列表
$< 比当前目标更新的已修改的当前依赖性名称
$^ 用空格分开的所有依赖性列表
@echo "Build complete"
power: $(OBJECTS)
$(CC) $(OPTIONS) -O $@ $^ -lm
@echo "The executable is in the power file."
main.o: main.h input.h compute.h
compute.o: compute.h
input.o: input.h
power.tar: Makefile $(HEADERS) $(SOURCES)
tar -cvf $@ $^
.PHONY:clean
clean:
rm -f *.o power
虚目标
允许强制执行某些在正常规则中不会发生的事件。
比如:可以通过设置一个虚目标生成多个可执行文件,若不使用虚目标,make就只能建立第一个目标。
.PHONY: rebuild-all
rebuild-all:prog1 prog2 prog3
prog1: prog1.o utils.o
cc -o prog1 prog1.o utils.o
prog2: prog2.o
cc -o $@ $^
prog3: prog3.o sort.o utils.o
cc -o $@ $^
常见虚目标列表
all 生成工程中所有可以执行者,通常是Makefile的第一个生成目标
$make all #使目标全部被执行
clean 删除make all生成的所有文件
install 在系统目录中安装工程项目生成的可执行文件和文档
uninstall 删除make install 安装的所有文件
$cat Makefile
INSTALLDIR=/home/embedclub/bin
install: client server
cp -f $^ $(INSTALLDIR)
rm -f *.o $^
cd $(INSTALLDIR); chmod 755 $^
uninstall:
cd $(INSTALLDIR); rm client server
client: client.o miscc.o rcopyc.o
gcc client.o miscc.o rcopyc.o -lnsl -o client \
client.o: client.c netc.h rcopy.h rcopy.h
gcc -c client.c
$make install
$make uninstall
一般性语法错误
1 缺少Tab键,可使用cat -t Makefile查看tab键位置
2 在连接符和换行符之间插入了空格,可使用cat -e Makefile查看换行符位置
显示Makefile中不正确的行 grep '\\[]$' Makefile
使用非标准的Makefile名称文件
$make -f prog1.makefile
从标准输入读取
$make -f -
显示Makefile中所执行命令的顺序
$make -n
制作工程文件的Makefile
一般的工程文件proc组成:
src:main.c fun1.c fun2.c
include:fun1.h fun2.h
Makefile
$cat Makefile
VPATH = src:include
all:test4 tar
.PHONY:all
test4:main.o fun1.o fun2.o
gcc main.o fun1.o fun2.o -o test4
main.o:main.c
gcc -c -linclude -o $@ $^
fun1.o:
fun2.o:
tar:
tar cvf test4.tar src include Makefile
.PHONY:clean
clean:
rm *.o test4
Makefile编写规则详解
Make命令与Makefile
Makefile文件内容
显示规则:说明了如何生成一个或多个目标文件(包括要生成的文件/文件的依赖文件/生成的命令)。
隐式规则
变量定义:一般是字符串,Makefile被执行时,其中变量都会拓展到相应的引用位。
文件指示: 1 在一个Makefile文件中引用另一个Makefile文件 (类似include)
2 根据某些情况指定makefile文件中的有效部分 (类似预编译#if)
3 定义一个多行的命令
注释: 注释符#(Makefile文件中需要用到#,可以使用\#转义)
注意:
makefile文件的文件名可以是其他名称,但要使用-f或--file指定
make工作执行步骤
1 读入所有makefile文件
2 读入被include包括的其他makefile文件
3 初始化文件中的变量
4 推导隐式规则,并分析所有规则
5 为所有目标文件创建依赖关系链
6 根据依赖关系,决定哪些目标要重新生成
7 执行生成命令
make参数 例子 作用
-f --file make -f makelinux 指定特定的makefile文件
-I --include-dir 在指定目录下寻找makefile文件
-n --just-print 只是显示->命令不执行命令
-s --silent 禁止命令的输出显示
makefile关键子 例子 作用
include include ../Make.defines 将别的makefile文件包含进来
include foo.make *.mk $(bar)
-include 不理会无法找到的文件
wildcard objects:= $(wildcard *.o) 让通配符在变量中展开,即让objects的值成为所有.o的文件名的集合
vpath <pattern> <directory> vpath %.h ../headers 为符合模式<pattern>的文件指定搜素目录
vpath %.c foo:bar
vpath blish 这两句一起用的话,表示.c结尾的文件先在foo,然后在bar,最后在blish中寻找
vpath <patterh> 清除符合模式<pattern>的文件的搜索目录
vpath 清除所有已设置好的文件搜索目录
定义环境变量 例子 作用
MAKEFILES 把此变量的值(其他makefile,多个文件用空格分隔)作为一个类似于include的动作
VPATH VPATH = src:../headers 让make根据路径寻找目标依赖文件,多个路径用:隔开
自动变量
$@ 目标集
$< 所有的依赖目标集
Makefile书写规则(里面的命令其实是shell命令)
通配符(定义一系列比较类似的文件)
* objects=*.o 所有的.o文件的集合
?
[]
~ ~/test 表示宿主主目录下test文件
文件搜寻
VPATH
vpath 关键字
伪目标
.PHONY .PHONY: clean 伪目标可以直接放在make后面像操作文件一样操作
make clean
多目标
静态模式
目标集合 目标集模式 目标集的二次定义
<targets ...>: <target-pattern>: <prereq-pattern>
<commands>
...
makefile文件的函数
filter
自动生成依赖关系
gcc -MM main.c //查找main.c文件包含的头文件,并生成依赖关系
.d文件的应用
使用命令
显示命令 @ehco (@避免输出命令,只输出命令执行结果)
执行命令
exec:
cd /home/hchen; pwd 第二条命令执行建立在第一条命令结果上
#伪目标同样可以存在依赖关系
.PHONY: cleanall cleanobj cleandiff
cleanall: cleanobj cleandiff
rm program
cleanobj:
rm *.o
cleaniff:
rm *.diff
#静态模式
objects=foo.o bar.o
all: $(objects)
$(objects): %.o: %.c
$(gcc) -c $(CFLAGS) $< -o $@ #展开后等价于
foo.o:foo.c
$(gcc) -c $(CFLAGS) foo.c -o foo.o
bar.o: bar.c
$(gcc) -c $(CFLAGS) bar.c -o bar.o
学习积累
源文件:ChessBoard.h ChessBoard.cpp Player.h Player.cpp main.cpp
依赖关系:
ChessBoard.cpp 包含ChessBoard.h
Player.cpp 包含 Player.h
main.cpp 包含ChessBoard.h 和 Player.h
Makefile:
gomoku: ChessBoard.o Player.o main.o
g++ -o gomoku ChessBoard.o Player.o main.o
#main.o: main.cpp Player.h ChessBoard.h
# g++ -c $^
#ChessBoard.o: ChessBoard.cpp ChessBoard.h
# g++ -c $^
#Player.o: Player.cpp Player.h
# g++ -c $^ #通过下面方法编译器可以自动推导
main.o: Player.h ChessBoard.h
ChessBoard.o: ChessBoard.h
Player.o: Player.h .PHONY: clean allclean
clean:
rm *.o *.gch
allclean:
rm *.o *.gch gomoku
Make和Makefile编写(详见GCC手册)的更多相关文章
- Yocto开发笔记之《Makefile编写》(QQ交流群:519230208)
开了一个交流群,欢迎爱好者和开发者一起交流,转载请注明出处. QQ群:519230208,为避免广告骚扰,申请时请注明 “开发者” 字样 =============================== ...
- linux 下C语言编程库文件处理与Makefile编写
做开发快3年了,在linux下编译安装软件算是家常便饭了.就拿gcc来说,都有不下10次了,可基本每次都会碰到些奇奇怪怪的问题.看来还是像vs.codeblocks这样的ide把人弄蠢了.便下定决心一 ...
- 单目录下多文件 makefile编写
makefile很久就接触过了,但是一直没怎么深入的去学习和总结:在项目中我也只是看看makefile或者修改部分语句,全部自己动手写的话还真没有:知识在于沉淀,这句说的非常好,所以现在把自己理解的东 ...
- 多目录下多文件 makefile编写
前面已经分享了单目录项下多文件的makefile的编写,现在来看看多目录下多文件makefile的编写: 在做项目时,一般文件都会分几个目录来存放:基本的是 include/ bin/ src/ ...
- linux --> Makefile编写
Makefile编写 单目录 测试程序在同一个文件中,共有func.h.func.c.main.c三个文件,Makefile写法如下所示: CC = gcc CFLAGS = -g -Wall mai ...
- Linux——makefile编写
以前对makefile的编写,限于刚开始接触,我都比较局限一些死板的格式,有时候就会显得有些繁琐.在进一步了解一些系统编译和链接的知识后,对makefile编写流程有了一些新的认识,所以来此梳理梳理, ...
- 如何将多个C文件链接在一起----Makefile编写及make指令
需使用GCC编译器,关于MinGW的安装指南:https://people.eng.unimelb.edu.au/ammoffat/teaching/20005/Install-MinGW.pdf 单 ...
- [动态库]动态库生成和使用以及Makefile编写
转自:https://www.cnblogs.com/ljtknowns/p/5647793.html 文件目录结构如下 1 dynamiclibapp.c 2 Makefile 3 comm/inc ...
- Makefile编写 一 *****
编译:把高级语言书写的代码转换为机器可识别的机器指令.编译高级语言后生成的指令虽然可被机器识别,但是还不能被执行.编译时,编译器检查高级语言的语法.函数与变量的声明是否正确.只有所有的语法正确.相关变 ...
随机推荐
- C语言实例代码
绘制余弦曲线和直线 #include #include int main() { double y; int x,m,n,yy; for(yy=0;yy<=20;yy++) {y=0.1*yy; ...
- c#读取文本文档实践1-File.ReadAllLines()
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.I ...
- 免费获得NOD32 半年、1年 激活码-14.08.12到期
地址: http://nod32.ruanmei.com/ 活动时间: 2014年8月6日 - 8月12日(全部送完将提前终止). 活动规则: 1.每台电脑限领1枚NOD32激活码: 2.领到的NOD ...
- SharePoint 2013 企业搜索架构示例
博客地址:http://blog.csdn.net/FoxDave 本文参考自微软官方的Chart,我们来看一下企业中对于不同规模SharePoint搜索的场的架构是什么样的. 对于搜索场的规模, ...
- 介绍NSURLSession网络请求套件
昨天翻译了一篇<NSURLSession的使用>的文章,地址:http://www.cnblogs.com/JackieHoo/p/4995733.html,原文是来自苹果官方介绍NSUR ...
- IOS界面切换
好吧!表示这几天要实现 phonegap 打开IOS原生界面,因此也查询了一些方案. 有如下几种: 第一种:navigationcontroller //进入下层 [self.navigationC ...
- js 弹出div窗口 可移动 可关闭 (转)
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/ ...
- SqlServer DateDiff函数 比较时间 (转)
DateDiff函数 是一个非常有用的函数,它可以为一些网页做一些特殊的效果. 我就曾用到它和一张'new'字样的图片 来区别网页显示的信息是否为最近的信息.例如:提示最近的通知,最近的新闻等等 ...
- 毕向东day01笔记--dos-jdk-jre-环境变量等
1.常用的dos命令,md,rd,dir,c:(进入C盘),del,set classpath 2.JDK和JRE之间的区别: JDK包含JER,JRE包含JVM. 3.环境变量的配置,静态配置--b ...
- eclipse安装hibernate