Makefile是干嘛的?
就像dockerfile之于docker,makefile之于make.....
make命令执行时,需要一个makefile文件,以告诉make命令需要怎么样的去编译和链接程序。
你在命令行敲一个make,会自动去寻找目录下的makefile文件并执行。
可能你在C程序中见到的比较多,但是其实其他语言也是可以用到。
举例一个简单的需求。如果你写一个api,在到测试环境之前,前端需要开发联调怎么办?不可能连你本地,因为你还要工作不停修改代码。一个方法是scp传到开发机,然后登陆开发机,在开发机编译出新的程序,然后,以后每次修改代码都需要这么来一遍,十分繁琐。
如果使用makefile,可以把这一系列的任务一键执行,类似于shell脚本。但是比起shell脚本的优点是,makefile可以把多个任务写在一个文件中,然后指定执行其中某一个,不用像shell脚本每个任务都需要编写一个。(当然shell脚本也能实现,但是相对较繁琐)
Makefile规则
makefile内部是你根据makefile语法规则,自己编写的一条条shell命令等。
基本格式
target ... : prerequisites ...
command
...
...
target
可以是一个object file(目标文件),也可以是一个执行文件,还可以是一个标签(label)。对于标签这种特性,在后续的“伪目标”章节中会有叙述。
prerequisites
生成该target所依赖的文件和/或target,没有就不需要写
command
该target要执行的命令(任意的shell命令)
下面我们看一个在golang
中的应用例子
.PHONY: test
devHost=root@127.0.0.1
test:
cd test && env GOOS=linux go build -ldflags="-s -w" -o testprog && upx -1 -k -v testprog
@rsync -e 'ssh -p 22' -c testprog $(devHost):/data/www/test/main
@ssh $(devHost) -p 22 "supervisorctl restart testprog";
rm -fr testprog*
这里执行的任务就是本地编译,把得到的二进制可执行文件用upx压缩之后通过rsync传到服务器,
然后重启服务器的supervisor使程序重启,最后删除本地生成的可执行文件。
这一系列的任务,在makefile编写完成之后,只需要执行一句
make test
就可以完成一系列任务,非常方便。
这里需要注意的几个地方:
.PHONY
.PHONY: build clean tool lint help
其作用是声明 build / clean / tool / lint / help 为伪目标,声明为伪目标会怎么样呢?
声明为伪目标后:在执行对应的命令时,make 就不会去检查是否存在 build / clean / tool / lint / help 其对应的文件,而是每次都会运行标签对应的命令
若不声明:恰好存在对应的文件,则 make 将会认为 xx 文件已存在,没有重新构建的必要了
@
如果你执行make
会发现,makefile中的每条命令都被打印到shell标准输出中,为什么?
因为make默认就是这么设置的,默认先打印每条命令再执行,这叫echo
在命令前加@作用就是不让你这条命令打印到标准输出。
有疑问加站长微信联系(非本文作者)