转载于陈皓大神
一个工程中的源文件不计其数,其按类型、功能、模块分别放在若干个目录中,makefile 定义了一系列的规则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,因为 makefile 就像一个 shell 脚本一样,其中也可以执行操作系统的命令。

Makefile 介绍

make 命令执行时,需要一个 Makefile 文件,以告诉 make 命令需要怎么样的去编译和链接程序。

Makefile 的规则

1
2
3
taget...: prerequisites ...
command
...

target也就是一个目标文件,可以是 Object File,也可以是执行文件。
prerequisites就是要生成哪个 target 所需要的文件或目标
command就是 make 需要执行的命令。(任意的 shell 命令)
这是一个文件的依赖关系,也就是说,target 这一个或多个的目标文件依赖于 prerequisites 中的文件,其生成规则定义在 command 中。

一个实例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
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

简单理解:冒号前面的就是目标文件(target),冒号后面就是依赖文件(prerequisites)。target 依赖 prerequisites。

make 是如何工作的

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

  • make 会在当前目录下找名字叫“Makefile”或“makefile”的文件
  • 如果找到,它会找文件中的第一个目标文件,如 edit
  • 如果 edit 不存在,或者 edit 所依赖的后面的文件要比 edit 这个文件新,那么,它就会执行后面所定义的命令(cc -o edit *.o)来生成 edit 这个文件
  • 如果 main.o 不存在,就会执行 command(cc -c main.c) 生成 main.o,依次类推,最后生成 edit

    其他

    clean 不是一个文件,它只不过是一个动作名字,其冒号后什么也没有,那么 make 就不会自动去找问价你的依赖性,也就不会自动执行其后所定义的命令。要执行其后的命令,就要在 make 命令后明显得指出这个 label 的名字。
    install 同 clean

    以 redis 为例

    通过源码安装 redis

    Downloading the source files

    1
    wget https://download.redis.io/redis-stable.tar.gz

    Compiling Redis

    1
    2
    3
    4
    5
    tar -xzvf redis-stable.tar.gz
    cd redis-stable
    make
    ### To install these binaries in /usr/local/bin
    make install
    摘抄一下 redis 的 Makefile 的 install
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    install: all
    @mkdir -p $(INSTALL_BIN)
    $(REDIS_INSTALL) $(REDIS_SERVER_NAME) $(INSTALL_BIN)
    $(REDIS_INSTALL) $(REDIS_BENCHMARK_NAME) $(INSTALL_BIN)
    $(REDIS_INSTALL) $(REDIS_CLI_NAME) $(INSTALL_BIN)
    $(REDIS_INSTALL) $(REDIS_CHECK_RDB_NAME) $(INSTALL_BIN)
    $(REDIS_INSTALL) $(REDIS_CHECK_AOF_NAME) $(INSTALL_BIN)
    @ln -sf $(REDIS_SERVER_NAME) $(INSTALL_BIN)/$(REDIS_SENTINEL_NAME)

    uninstall:
    rm -f $(INSTALL_BIN)/{$(REDIS_SERVER_NAME),$(REDIS_BENCHMARK_NAME),$(REDIS_CLI_NAME),$(REDIS_CHECK_RDB_NAME),$(REDIS_CHECK_AOF_NAME),$(REDIS_SENTINEL_NAME)}
    虽然看不懂,但是我猜应该是把 redis 的一些可执行文件 ln(创建文件链接) 到 /usr/local/bin