跟我一起写makefile入门笔记

Table of Contents

 

一、makefile是如何工作的(依赖关系)

二、文件搜寻 VPATH

三、makefile函数的使用:

1、$(patsubst%.c,%.o,x.c.c bar.c) 把.c字符串换成.o的字符串

2、files:=$(foreach n,$(names),$(n).o) 把name文件名中含有.o 名字文件提取出来给files

         awk刷选文件及文件夹



 

一、makefile是如何工作的(依赖关系)

在默认的方式下,也就是我们只输入make命令。那么,

1.make会在当前目录下找名字叫“Makefile”或“makefile”的文件。
2.如果找到,它会找文件中的第一个目标文件(target),在上面的例子中,他会找到“edit”这个文件,并把这个文件作为最终的目标文件。
3.如果edit文件不存在或是edit所依赖的后面的.o文件的文件修改时间要比edit这个文件新,那么,他就会执行后面所定义的命令来生成edit这个文件。
4.如果edit所依赖的.o文件也不存在,那么make会在当前文件中找目标为.o文件的依赖性,如果找到则再根据那一个规则生成.o文件。(这有点像一个堆栈的过程)
5.当然,你的C文件和H文件是存在的啦,于是make会生成.o文件,然后再用.o文件生成make的终极任务,也就是执行文件edit了。

这就是整个make的依赖性,make会一层又一层地去找文件的依赖关系,直到最终编译出第一个目标文件。在找寻的过程中,如果出现错误,比如最后被依赖的文件找不到,那么make就会直接退出,并报错,而对于所定义的命令的错误,或是编译不成功,make根本不理。make只管文件的依赖性,即,如果在我找了依赖关系之后,冒号后面的文件还是不在,那么对不起,我就
不工作啦。

通过上述分析,我们知道,像clean这种,没有被第一个目标文件直接或间接关联,那么它后面所定义的命令将不会被自动执行,不过,我们可以显示要make执行。即命令——make clean,以此来清除所有的目标文件,以便重编译。

于是在我们编程中,如果这个工程已被编译过了,当我们修改了其中一个源文件,比如file.c,那么根据我们的依赖性,我们的目标file.o会被重编译(也就是在这个依性关系后面所定义的命令),于是file.o的文件也是最新的啦,于是file.o的文件修改时间要比edit要新,所以edit也会被重新链接了(详见edit目标文件后定义的命令)。而如果我们改变了command.h,那么,kdb.o、command.o和files.o都会被重编译,并且,edit会被重链接。

重点总结:

当已经编译完后的工程,如果修改了某个文件后,再次make, 自会重新编译修改过的文件,而不会修改其它文件,原因是:目标文件是否更新要看依赖文件,如果依赖文件更新了,目标文件也要更新

 

二、文件搜寻 VPATH

在一些大的工程中,有大量的源文件,我们通常的做法是把这许多的源文件分类,并存放在不同的目录中。所以,当make需要去找寻文件的依赖关系时,你可以在文件前加上路径,但最好的方法是把一个路径告诉make,让make在自动去找。

Makefile文件中的特殊变量“VPATH”就是完成这个功能的,如果没有指明这个变量,make只会在当前的目录中去找寻依赖文件和目标文件。如果定义了这个变量,那么,make就会在当当前目录找不到的情况下,到所指定的目录中去找寻文件了。

    VPATH = src:../headers

上面的的定义指定两个目录,“src”和“../headers”,make会按照这个顺序进行搜索。目录由“冒号”分隔。(当然,当前目录永远是最高优先搜索的地方)

另一个设置文件搜索路径的方法是使用make的“vpath”关键字(注意,它是全小写的),这不是变量,这是一个make的关键字,这和上面提到的那个VPATH变量很类似,但是它更为灵活。它可以指定不同的文件在不同的搜索目录中。这是一个很灵活的功能。它的使用方法有三种:

 为符合模式<pattern>的文件指定搜索目录<directories>。

    1、vpath <pattern> <directories>

     清除符合模式<pattern>的文件的搜索目录。

    2、vpath <pattern>

      清除所有已被设置好了的文件搜索目录。

    3、vpath

vapth使用方法中的<pattern>需要包含“%”字符。“%”的意思是匹配零或若干字符,例如,“%.h”表示所有以“.h”结尾的文件。<pattern>指定了要搜索的文件集,而<directories>则指定了<pattern>的文件集的搜索的目录。例如:

    vpath %.h ../headers

该语句表示,要求make在“../headers”目录下搜索所有以“.h”结尾的文件。(如果某文件在当前目录没有找到的话)

我们可以连续地使用vpath语句,以指定不同搜索策略。如果连续的vpath语句中出现了相同的<pattern>,或是被重复了的<pattern>,那么,make会按照vpath语句的先后顺序来执行搜索。如:

    vpath %.c foo
    vpath %   blish
    vpath %.c bar

其表示“.c”结尾的文件,先在“foo”目录,然后是“blish”,最后是“bar”目录。

    vpath %.c foo:bar
    vpath %   blish

而上面的语句则表示“.c”结尾的文件,先在“foo”目录,然后是“bar”目录,最后才是“blish”目录。
 

三、makefile函数的使用:

函数调用,很像变量的使用,也是以$来标识的,其法如下

$(<function><arguments>)

或是:

${<function><arguments>}

函数调用以$开头  <arguments>为函数的参数,

1、$(patsubst%.c,%.o,x.c.c bar.c) 把.c字符串换成.o的字符串

$(patsubst<pattern>,<replacement>,<text>)

 

2、files:=$(foreach n,$(names),$(n).o) 把name文件名中含有.o 名字文件提取出来给files

$(foreach<var>,<list>,<text>) 

awk刷选文件及文件夹

参考https://blog.csdn.net/shenlong1356/article/details/91554209

取出DEBUG目录

DEBUG=$(shell ls -l | grep ^d | awk '{if($9 == "debug") print $9}')

待更新

发布了135 篇原创文章 · 获赞 112 · 访问量 6万+

猜你喜欢

转载自blog.csdn.net/shenlong1356/article/details/91492772
今日推荐