Makefile12--自动生成依赖关系(中)
学习自狄泰软件学院唐佐临老师Makefile课程,文章中图片取自老师的PPT,仅用于个人笔记。
实验1 :当前目录中没有 test.txt文件,makefile 中有对应规则
实验2 :makefile 中 没有 include关键字后面 所对应的目标,当前目录也没有对应文件
实验3:当前目录中没有 test.txt文件,makefile 中有对应规则,并在规则中创建 test.txt .执行两次 make all
实验4 : 在实验3的基础上,在 test.txt文件中随意添加一段字符串,然后make(不是make all)
实验5 :规则中的每一个命令默认是在一个新的进程中执行
实验6:规则中的每一个命令默认是在一个新的进程中执行,使用接续符让所有命令在同一进程中执行。
实验7:包含所有的依赖文件,并得到依赖文件,最后在依赖文件中填充依赖信息。
实验8:-include :屏蔽了警告提示信息
在makefile中, include关键字后面可以跟如下三类:
foo.mk :具体的想要包含的文件名
*.mk :通配符的方式,指包含当前目录中所有的.mk文件,将当前目录中所有的.mk文件中的的内容全部搬到当前的makefile中来。
$(var) :某个变量值,意思是 包含这个变量所代表的那一个文件的内容
实验1 :当前目录中没有 test.txt文件,makefile 中有对应规则
.PHONY : all
include test.txt
all :
@echo "this is all"
test.txt :
@echo "test.txt"
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ make all
makefile:4: test.txt: No such file or directory
this is test.txt
this is all
mhr@ubuntu:~/work/makefile1$
实验2 :makefile 中 没有 include关键字后面 所对应的目标,当前目录也没有对应文件
.PHONY : all
include test.txt
all :
@echo "this is all"
/makefile1$ make all
makefile:4: test.txt: No such file or directory
make: *** No rule to make target 'test.txt'. Stop.
mhr@ubuntu:~/work/makefile1$
实验3:当前目录中没有 test.txt文件,makefile 中有对应规则,并在规则中创建 test.txt .执行两次 make all
.PHONY : all
include test.txt
all :
@echo "this is all"
test.txt :
@echo "test.txt"
#创建 test.txt
@touch test.txt
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ ls
func.c func.h func.o hello.out main.c main.o makefile
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ make all
makefile:5: test.txt: No such file or directory
this is test.txt
this is all
mhr@ubuntu:~/work/makefile1$ ls
func.c func.h func.o hello.out main.c main.o makefile test.txt
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ make all
this is all
mhr@ubuntu:~/work/makefile1$
第一次执行 make all 时候,执行include关键字 所对应的规则,打印字符串病创建 test.txt文件。结果在当前目录生成了test.txt文件。
第二次执行 make all的时候,include关键字找到了 对应的test,txt文件,于是将 test.txt文件里面的内容拷贝过来了,就没必要再去执行 对应的规则了。
实验4 : 在实验3的基础上,在 test.txt文件中随意添加一段字符串,然后make(不是make all)
test.txt
other :
@echo "this is $@"
makefile
.PHONY : all
include test.txt
all :
@echo "this is all"
test.txt :
@echo "test.txt"
#创建 test.txt
@touch test.txt
mhr@ubuntu:~/work/makefile1$ make
this is other
mhr@ubuntu:~/work/makefile1$
说明:
此处执行的是 make,并不是 make all , make 会去执行 makefile 中最顶层的规则,而此时 makefie中最顶层的规则 已经不是 all 了,因为 include关键字已经将 test.txt 中的内容拷贝到了nakefile 的最顶层了,所以此时最顶层的规则是 test.txt中的内容:
other :
@echo "this is $@"
实验5 :规则中的每一个命令默认是在一个新的进程中执行
makefile
.PHONY : all
all :
mkdir test
cd test
mkdir subtest
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ make
mkdir test
cd test
mkdir subtest
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ ls -l
total 48
-rw-rw-r-- 1 mhr mhr 93 Dec 27 04:13 func.c
-rw-rw-r-- 1 mhr mhr 83 Dec 27 05:58 func.h
-rw-rw-r-- 1 mhr mhr 1552 Dec 27 05:58 func.o
-rwxrwxr-x 1 mhr mhr 8656 Dec 27 05:58 hello.out
-rw-rw-r-- 1 mhr mhr 81 Dec 27 04:13 main.c
-rw-rw-r-- 1 mhr mhr 1368 Dec 27 05:58 main.o
-rw-rw-r-- 1 mhr mhr 62 Dec 27 07:02 makefile
drwxrwxr-x 2 mhr mhr 4096 Dec 27 07:03 subtest
drwxrwxr-x 2 mhr mhr 4096 Dec 27 07:03 test
-rw-rw-r-- 1 mhr mhr 28 Dec 27 06:57 test.txt
mhr@ubuntu:~/work/makefile1$
结果是分别创建了test 和 subtest文件夹,并不是我们想要的结果。就是因为makefile 中 规则中的每一个命令默认是在一个新的进程中执行,上面规则中的三个命令是在三个进程中执行的 没有连贯性
进程x :mkdir test
进程结束
进程y: cd test
进程结束
进程z:mkdir subtest
进程结束
实验6:规则中的每一个命令默认是在一个新的进程中执行,使用接续符让所有命令在同一进程中执行。
注意 这里的 \ 仅仅是格式的连接符,表示这些代码在同一行。而 ;才是makefile的命令接续符,表示这些命令在同一个进程中执行。
.PHONY : all
all :
set -e; \
mkdir test; \
cd test; \
mkdir subtest
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ make
set -e; \
mkdir test; \
cd test; \
mkdir subtest
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ ll
total 52
drwxrwxr-x 3 mhr mhr 4096 Dec 27 07:10 ./
drwxrwxr-x 4 mhr mhr 4096 Dec 15 03:18 ../
-rw-rw-r-- 1 mhr mhr 93 Dec 27 04:13 func.c
-rw-rw-r-- 1 mhr mhr 83 Dec 27 05:58 func.h
-rw-rw-r-- 1 mhr mhr 1552 Dec 27 05:58 func.o
-rwxrwxr-x 1 mhr mhr 8656 Dec 27 05:58 hello.out*
-rw-rw-r-- 1 mhr mhr 81 Dec 27 04:13 main.c
-rw-rw-r-- 1 mhr mhr 1368 Dec 27 05:58 main.o
-rw-rw-r-- 1 mhr mhr 79 Dec 27 07:07 makefile
drwxrwxr-x 3 mhr mhr 4096 Dec 27 07:10 test/
-rw-rw-r-- 1 mhr mhr 28 Dec 27 06:57 test.txt
mhr@ubuntu:~/work/makefile1$ cd test/
mhr@ubuntu:~/work/makefile1/test$ ll
total 12
drwxrwxr-x 3 mhr mhr 4096 Dec 27 07:10 ./
drwxrwxr-x 3 mhr mhr 4096 Dec 27 07:10 ../
drwxrwxr-x 2 mhr mhr 4096 Dec 27 07:10 subtest/
mhr@ubuntu:~/work/makefile1/test$
实验7:得到依赖文件,并包含所有的依赖文件
.PHONY : all clean
MKDIR := mkdir
RM := rm -fr
CC := gcc
#将所有当前目录中的.c作为后缀的文件的文件名作为列表拿到,并保存到变量 SRCS 中
SRCS := $(wildcard *.c)
#变量的值替换 替换后缀
DEPS := $(SRCS:.c=.dep)
#包含依赖文件,因为依赖文件中包含目标的部分依赖
#此时当前目录并没有依赖文件,所以利用 include关键字特性,寻找依赖文件对应规则,在规则中我们可以自己生成依赖文件。
include $(DEPS)
all :
@echo "all"
#针对目录结构的模式规则
%.dep : %.c
@echo "Creating $@ ..."
@set -e; \
$(CC) -MM -E $^ | sed 's,\(.*\)\.o[ :]*,objs/\1.o : ,g' > $@
clean :
$(RM) $(DEPS)
mhr@ubuntu:~/work/makefile1$ make all
makefile:11: main.dep: No such file or directory
makefile:11: func.dep: No such file or directory
Creating func.dep ...
Creating main.dep ...
all
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ ll
total 52
drwxrwxr-x 2 mhr mhr 4096 Dec 27 08:07 ./
drwxrwxr-x 4 mhr mhr 4096 Dec 15 03:18 ../
-rw-rw-r-- 1 mhr mhr 93 Dec 27 04:13 func.c
-rw-rw-r-- 1 mhr mhr 28 Dec 27 08:07 func.dep
-rw-rw-r-- 1 mhr mhr 83 Dec 27 05:58 func.h
-rw-rw-r-- 1 mhr mhr 1552 Dec 27 05:58 func.o
-rwxrwxr-x 1 mhr mhr 8656 Dec 27 05:58 hello.out*
-rw-rw-r-- 1 mhr mhr 81 Dec 27 04:13 main.c
-rw-rw-r-- 1 mhr mhr 28 Dec 27 08:07 main.dep
-rw-rw-r-- 1 mhr mhr 1368 Dec 27 05:58 main.o
-rw-rw-r-- 1 mhr mhr 285 Dec 27 08:07 makefile
mhr@ubuntu:~/work/makefile1$
从结果看 已经生成了两个依赖文件:main.dep,func.dep,内容如下
main.dep
objs/main.o : main.c func.h
func.dep
objs/func.o : func.c func.h
实验8:-include :屏蔽了警告提示信息
.PHONY : all clean
MKDIR := mkdir
RM := rm -fr
CC := gcc
SRCS := $(wildcard *.c)
DEPS := $(SRCS:.c=.dep)
include $(DEPS)
all :
@echo "all"
%.dep : %.c
@echo "Creating $@ ..."
@set -e; \
$(CC) -MM -E $^ | sed 's,\(.*\)\.o[ :]*,objs/\1.o : ,g' > $@
clean :
$(RM) $(DEPS)
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ make all
Creating func.dep ...
Creating main.dep ...
all
mhr@ubuntu:~/work/makefile1$
冰红茶兑滴水: 什么书
CSDN-Ada助手: 哇, 你的文章质量真不错,值得学习!不过这么高质量的文章, 还值得进一步提升, 以下的改进点你可以参考下: (1)增加条理清晰的目录;(2)使用更多的站内链接;(3)增加除了各种控件外,文章正文的字数。
CSDN-Ada助手: 云原生时代,Nginx是否还是很重要,还是说云原生里的网关能把Nginx消灭掉?
CSDN-Ada助手: 如何衡量企业网络的性能指标?
CSDN-Ada助手: TCP 协议中的 FIN 握手是什么?有什么作用?