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$ 
发布了192 篇原创文章 · 获赞 100 · 访问量 8万+

猜你喜欢

转载自blog.csdn.net/LinuxArmbiggod/article/details/103739215