跟我一起写Makefile细节总结学习笔记

2023-09-20 15:39:17

跟我一起写Makefile细节总结学习笔记

第一,二章

此篇仅为方便查阅记忆,详细的请看seisman/how-to-write-makefile: 跟我一起写Makefile重制版 (github.com)

Makefile介绍

,,Tab键,使用变量,-减号,

规则

target ... : prerequisites ...
	recipe
	...
	...

原始Makefile

edit : main.o kbd.o command.o display.o \
insert.o search.o files.o utils.o
	cc -o edit main.o kbd.o command.o display.o \
insert.o search.o files.o utils.o
main.o : main.c defs.h
	cc -c main.c
kbd.o : kbd.c defs.h command.h
	cc -c kbd.c
command.o : command.c defs.h command.h
	cc -c command.c
display.o : display.c defs.h buffer.h
	cc -c display.c
insert.o : insert.c defs.h buffer.h
	cc -c insert.c
search.o : search.c defs.h buffer.h
	cc -c search.c
files.o : files.c defs.h buffer.h command.h
	cc -c files.c
utils.o : utils.c defs.h
	cc -c utils.c
clean :
	rm edit main.o kbd.o command.o display.o \
		insert.o search.o files.o utils.o

稍微简化后的

objects = main.o kbd.o command.o display.o \
	insert.o search.o files.o utils.o
edit : $(objects)
	cc -o edit $(objects)
$(objects) : defs.h
kbd.o command.o files.o : command.h
display.o insert.o search.o files.o : buffer.h
.PHONY : clean
clean :
	-rm edit $(objects)

注意

命令的Tab键位不可用空个代替。

在我们的 makefile 中以 $(objects) 的方式来使用这个变量

反斜杠(\ )是换行符的意思。

.PHONY 表示 clean 是一个“伪目标”。

一个小减号的意思就是,也 许某些文件出现问题,但不要管,继续做后面的事

不成文的规矩是——“clean 从来都是放在文件的最 后”。

Makefile 里主要包含了五个东西:

显式规则、隐式规则、变量定义、指令和注释。

  1. 显式规则。显式规则说明了如何生成一个或多个目标文件。这是由 Makefile 的书写者明显指出要 生成的文件、文件的依赖文件和生成的命令。
  2. 隐式规则。由于我们的 make 有自动推导的功能,所以隐式规则可以让我们比较简略地书写 Makefile,这是由 make 所支持的。 8 Chapter 2. makefile 介绍 , 发行版本 1.0
  3. 变量的定义。在 Makefile 中我们要定义一系列的变量,变量一般都是字符串,这个有点像你 C 语 言中的宏,当 Makefile 被执行时,其中的变量都会被扩展到相应的引用位置上。
  4. 指令。其包括了三个部分,一个是在一个 Makefile 中引用另一个 Makefile,就像 C 语言中的 include 一样;另一个是指根据某些情况指定 Makefile 中的有效部分,就像 C 语言中的预编译 #if 一样; 还有就是定义一个多行的命令。有关这一部分的内容,我会在后续的部分中讲述。
  5. 注释。Makefile 中只有行注释,和 UNIX 的 Shell 脚本一样,其注释是用 # 字符,这个就像 C/C++ 中的 // 一样。如果你要在你的 Makefile 中使用 # 字符,可以用反斜杠进行转义,如:# 。 最后,还值得一提的是,在 Makefile 中的命令,必须要以 Tab 键开始。

书写规则

在规则中使用通配符

另给一个变量使用通配符的例子:

  1. 列出一确定文件夹中的所有 .c 文件。

    objects := $(wildcard *.c) 
    
  2. 列出 (1) 中所有文件对应的 .o 文件,在(3)中我们可以看到它是由 make 自动编译出的:

     $(patsubst %.c,%.o,$(wildcard *.c)) 
    
  3. 由 (1)(2) 两步,可写出编译并链接所有 .c 和 .o 文件

    objects := $(patsubst %.c,%.o,$(wildcard *.c)) 
    foo : $(objects) 
    	cc -o foo $(objects)文件搜寻
        
    

文件搜寻

vpath <pattern> <directories>

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

vpath <pattern>

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

vpath

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

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

vpath %.h ../headers

伪目标

当然,为了避免和文件重名的这种情况,我们可以使用一个特殊的标记“.PHONY”来显式地指明 一个目标是“伪目标”,向 make 说明,不管是否有这个文件,这个目标就是“伪目标”。 .PHONY : clean 只要有这个声明,不管是否有“clean”文件,要运行“clean”这个目标,只有“make clean”这样。 于是整个过程可以这样写:

.PHONY : clean
clean :
		rm *.o temp

多目标(自动变量)

bigoutput littleoutput : text.g
	generate text.g -$(subst output,,$@) > $@

等价于

bigoutput : text.g
	generate text.g -big > bigoutput
littleoutput : text.g
	generate text.g -little > littleoutput

静态模式

objects = foo.o bar.o
	all: $(objects)
$(objects): %.o: %.c
	$(CC) -c $(CFLAGS) $< -o $@

上面的例子中,指明了我们的目标从 $object 中获取,%.o 表明要所有以 .o 结尾的目标,也就是 foo.o bar.o ,也就是变量 $object 集合的模式,而依赖模式 %.c 则取模式 %.o 的 % ,也就是 foo bar ,并为其加下 .c 的后缀,于是,我们的依赖目标就是 foo.c bar.c 。而命令中的 $< 和 @ 则是自动化变量, @ 则是自动化 变量, @则是自动化变量,< 表示第一个依赖文件,$@ 表示目标集(也就是“foo.o bar.o”)。于是,上面的规则展开后等价 于下面的规则:

foo.o : foo.c
	$(CC) -c $(CFLAGS) foo.c -o foo.o
bar.o : bar.c
	$(CC) -c $(CFLAGS) bar.c -o bar.o

试想,如果我们的 %.o 有几百个,那么我们只要用这种很简单的“静态模式规则”就可以写完一堆 规则,实在是太有效率了。“静态模式规则”的用法很灵活,如果用得好,那会是一个很强大的功能。再 看一个例子:

files = foo.elc bar.o lose.o
    
$(filter %.o,$(files)): %.o: %.c
	$(CC) -c $(CFLAGS) $< -o $@
$(filter %.elc,$(files)): %.elc: %.el
	emacs -f batch-byte-compile $<

( f i l t e r (filter %.o, (filter(files)) 表示调用 Makefile 的 filter 函数,过滤“$files”集,只要其中模式为“%.o”的 内容。其它的内容,我就不用多说了吧。这个例子展示了 Makefile 中更大的弹性。

自动生成依赖性

在 Makefile 中,我们的依赖关系可能会需要包含一系列的头文件,比如,如果我们的 main.c 中有一 句 #include “defs.h” ,那么我们的依赖关系应该是:

main.o : main.c defs.h

但是,如果是一个比较大型的工程,你必需清楚哪些 C 文件包含了哪些头文件,并且,你在加入或 删除头文件时,也需要小心地修改 Makefile,这是一个很没有维护性的工作。为了避免这种繁重而又容 易出错的事情,我们可以使用 C/C++ 编译的一个功能。大多数的 C/C++ 编译器都支持一个“-M”的 选项,即自动找寻源文件中包含的头文件,并生成一个依赖关系。例如,如果我们执行下面的命令:

cc -M main.c

其输出是:

main.o : main.c defs.h

于是由编译器自动生成的依赖关系,这样一来,你就不必再手动书写若干文件的依赖关系,而由编 译器自动生成了。需要提醒一句的是,如果你使用 GNU 的 C/C++ 编译器,你得用 -MM 参数,不然,-M 参数会把一些标准库的头文件也包含进来。

gcc -M main.c 的输出是

main.o: main.c defs.h /usr/include/stdio.h /usr/include/features.h \
	/usr/include/sys/cdefs.h /usr/include/gnu/stubs.h \
	/usr/lib/gcc-lib/i486-suse-linux/2.95.3/include/stddef.h \
	/usr/include/bits/types.h /usr/include/bits/pthreadtypes.h \
	/usr/include/bits/sched.h /usr/include/libio.h \
	/usr/include/_G_config.h /usr/include/wchar.h \
	/usr/include/bits/wchar.h /usr/include/gconv.h \
	/usr/lib/gcc-lib/i486-suse-linux/2.95.3/include/stdarg.h \
	/usr/include/bits/stdio_lim.h

gcc -MM main.c 的输出则是:

main.o: main.c defs.h
更多推荐

WebLOAD: 一站式性能测试工具

WebLOAD是一款一站式前端性能测试工具,对测试人员来说使用非常方便。它可以帮助前端工程师和测试快速对网页进行性能测试和优化,提高网页加载速度,减少页面卡顿和闪烁。WebLOAD的特点、使用指南以及企业实际使用中的案列。WebLOAD的特点功能丰富:WebLOAD集成了众多前端性能测试工具,如前端性能分析、首屏时间预

算法刷题 week1 找出数组中重复的数字&不修改数组找出重复的数字

目录week11.找出数组中重复的数字题目数据范围样例题解(数组遍历)O(n)2.不修改数组找出重复的数字题目数据范围样例题解(分治,抽屉原理)O(nlogn)week11.找出数组中重复的数字题目给定一个长度为n的整数数组nums,数组中所有的数字都在0∼n−1的范围内。数组中某些数字是重复的,但不知道有几个数字重复

MySQL

当谈到数据库管理系统时,MySQL通常是许多开发者和数据专业人士的首选。MySQL是一种开源的关系型数据库管理系统,广泛用于Web应用程序、企业应用程序和各种数据驱动的应用。在这篇博客中,我们将深入探讨MySQL的重要性、用途、基本概念以及如何开始使用它。什么是MySQL?MySQL是一个关系型数据库管理系统(RDBM

3.3 DLL注入:突破会话0强力注入

Session是Windows系统的一个安全特性,该特性引入了针对用户体验提高的安全机制,即拆分Session0和用户会话,这种拆分Session0和Session1的机制对于提高安全性非常有用,这是因为将桌面服务进程,驱动程序以及其他系统级服务取消了与用户会话的关联,从而限制了攻击者可用的攻击面。由于DLL注入在Se

警惕!多本SCI/SSCI被剔除,9月SCI/SSCI期刊目录已更新~(附下载)

【SciencePub学术】2023年9月20日,科睿唯安更新了WebofScience核心期刊目录。继上次SCI期刊目录和SSCI期刊目录更新之后,本次9月更新共有9本期刊发生变动:•SCIE:有3本期刊不再被SCIE期刊目录收录(EditorialDe-listing/ProductionDe-listing),1

避雷!这9本期刊已被剔除!9月SCI/SSCI目录已更新(附2023年WOS历次更新全目录)

2023年9月20日,科睿唯安更新了WebofScience核心期刊目录。此次更新后SCIE期刊目录共包含9490本期刊,SSCI期刊目录共包含3552本期刊。此次SCIE&SSCI期刊目录更新,与上次更新(2023年8月)相比,共有9本期刊发生变动。其中有3本SCIE期刊因不满足收录的质量标准或未从出版社接收到相关内

基于Java+SpringBoot+Vue+echarts校园资料分享平台设计和实现

博主介绍:✌全网粉丝30W+,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌🍅文末获取源码联系🍅👇🏻精彩专栏推荐订阅👇🏻不然下次找不到哟2022-2024年最全的计算机软件毕业设计选题

应用开发平台能力扩展——集成echarts组件实现图表展现能力

背景图表展现能力是平台需具备的基础能力,目前echarts是最佳选择。在早期版本的图表中,不同的图表样式,需要不同的数据格式,需要进行复杂封装才能易于使用。百度官方也意识到这个问题,在echarts4.0版本提供了dataset属性支持,提供了统一的数据格式,也曾考虑基于这一新特性将常用图表进行封装。后来,发现了饿了么

基于Java+SpringBoot+Vue+echarts健身房管理系统设计和实现

博主介绍:✌全网粉丝30W+,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌🍅文末获取源码联系🍅👇🏻精彩专栏推荐订阅👇🏻不然下次找不到哟2022-2024年最全的计算机软件毕业设计选题

autosar 诊断入门

AUTOSAR(汽车开放系统架构)是一个国际汽车行业的开放和标准化的软件架构。它的主要目标是为了创建一种独立于硬件的软件架构,以提高汽车电子系统的模块化和可重用性。AUTOSAR架构主要分为两个部分:AUTOSARRuntimeEnvironment(RTE)和AUTOSARSoftwareComponents(SWC

嵌入式学习笔记(31)异常向量表的编程处理

6.5.1像内存一样去访问异常向量表(1)S5PV210的异常向量表可以改变(在CP15协处理器中),以适应操作系统的需求。但是目前系统刚启动,此时DRAM尚未初始化,程序哦都市在iSRAM中运行。210在iSRAM中设置了异常向量表,供暂时性使用。(2)查210的iROMapplicationnote文档中iRAM的

热文推荐