背景
- 会不会写makefile,从一个侧面说明了一个人是否具备完成大型工程的能力
- 一个工程中的源文件不计数,其按类型、功能、模块分别放在若干个目录中,makefile定义了一系列的规则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,甚至于进行更复杂的功能操作
- makefile带来的好处就是——“自动化编译”,一旦写好,只需要一个make命令,整个工程完全自动编译,极大的提高了软件开发的效率。
- make是一个命令工具,是一个解释makefile中指令的命令工具,一般来说,大多数的IDE都有这个命令,比如:Delphi的make,Visual C++的nmake,Linux下GNU的make。可见,makefile都成为了一种在工程方面的编译方法。
- make是一条命令,makefile(或者Makefile)是一个文件,两个搭配使用,完成项目自动化构建。
基本语法介绍
Makefile 是用于构建自动化构建流程的文件,主要用于编译代码。Makefile 是由一系列规则和命令组成的脚本文件,控制文件的生成和更新。其语法主要包括目标、依赖和命令三个部分。以下是基本语法介绍:
基本结构
target: dependencies
[TAB] command
- target:目标文件或目标名字,通常是要生成的文件名,也可以是一个标签。
- dependencies:目标所依赖的文件或其他目标。如果这些依赖项发生变化,目标会被重新生成。
- command:为了生成目标文件需要执行的命令。注意,这里的命令行必须以 [TAB] 开头,而不能用空格。
例子
假设你有一个简单的 C 程序 hello.c
,你希望通过 Makefile
来编译这个程序生成 hello
可执行文件。你的 Makefile
可以是这样的:
hello: hello.o
gcc -o hello hello.o
hello.o: hello.c
gcc -c hello.c
特殊符号和变量
$@:表示目标文件的名字。
- $<:表示第一个依赖文件的名字。
- $^:表示所有依赖文件的名字,空格分隔。
例如,下面的 Makefile
等价于上面的例子
hello: hello.o
gcc -o $@ $^
hello.o: hello.c
gcc -c $<
伪目标
伪目标是指不会生成实际文件的目标,通常用于执行一些命令,如清理生成的文件。使用 .PHONY
来定义伪目标:
.PHONY: clean
clean:
rm -f hello hello.o
变量
你可以在 Makefile 中定义变量,用于简化代码管理。比如:
CC = gcc
CFLAGS = -Wall
hello: hello.o
$(CC) -o hello hello.o
hello.o: hello.c
$(CC) $(CFLAGS) -c hello.c
条件判断
Makefile
也支持条件判断:
ifeq ($(CC),gcc)
CFLAGS += -O2
endif
-
ifeq ($(CC),gcc)
:ifeq
是Makefile
中的一个条件判断语句,类似于其他编程语言中的if
语句。$(CC)
是一个变量引用,它表示当前的编译器(结合上面我们定义的变量,就是gcc
)。gcc
是GNU C
编译器的名字。- 这个条件的意思是:“如果变量
CC
的值等于gcc
,则执行接下来的代码块。
-
CFLAGS += -O2
:CFLAGS
是一个常见的Makefile
变量,表示编译器的标志或选项(C Compiler Flags)。+=
是追加操作符,表示将右边的内容追加到CFLAGS
变量的末尾。-O2
是一个编译选项,表示进行优化级别为 2 的优化(中等优化,适用于大多数情况)。- 这行代码的意思是:“如果编译器是
gcc
,那么将-O2
这个优化选项追加到CFLAGS
中。”
-
endif
:endif
表示条件判断的结束,对应前面的ifeq
。
注释
使用 # 来添加注释,注释内容从 # 开始,直到行尾:
# 这是一个注释