编写 Makefile 的过程如下:
- 定义变量:定义编译器、编译选项等变量。
- 定义目标文件:定义目标文件名。
- 定义源文件:定义所有的源文件。
- 定义规则:定义将每个源文件编译成目标文件的规则。
- 定义伪目标:定义清除目标文件和其他中间文件的伪目标。
下面是一个示例 Makefile,假设每个目录都有一个以 .c
结尾的源文件需要被编译成对应的 .o
文件:
CC = gcc
CFLAGS = -Wall -Wextra -Werror
SRCDIR = src
OBJDIR = obj
TARGET = myapp
SOURCES := $(wildcard $(SRCDIR)/**/*.c)
OBJECTS := $(patsubst $(SRCDIR)/%.c, $(OBJDIR)/%.o, $(SOURCES))
$(TARGET): $(OBJECTS)
$(CC) $(CFLAGS) $^ -o $@
$(OBJDIR)/%.o: $(SRCDIR)/%.c
$(CC) $(CFLAGS) -c $< -o $@
.PHONY: clean
clean:
rm -rf $(OBJDIR)/*.o $(TARGET)
CC
定义了使用的编译器,这里使用 gcc。CFLAGS
定义了编译选项,这里开启了一些警告选项。SRCDIR
定义了源文件所在的目录。OBJDIR
定义了中间文件和目标文件所在的目录。TARGET
定义了最终生成的可执行文件名。SOURCES
是所有源文件的列表,使用了通配符**
可以递归查找子目录中的源文件。OBJECTS
是所有目标文件的列表,使用了 patsubst 函数将源文件路径替换成目标文件路径。$(TARGET): $(OBJECTS)
表示目标文件依赖于所有的中间文件,可以通过在命令行输入$ make myapp
来生成最终的可执行文件。$(OBJDIR)/%.o: $(SRCDIR)/%.c
表示规则,将所有一个$(SRCDIR)
目录下的.c
文件编译成对应的.o
文件,并存储到$(OBJDIR)
目录下。.PHONY: clean
表示 clean 是一个伪目标,不是真正的文件,可以通过在命令行输入$ make clean
来删除所有的中间文件和目标文件。