Makefile的使用

本文详细介绍了Makefile的必要性、高效编译原理、内置函数的使用以及如何一步步完善Makefile,从简单示例到支持头文件检测和自动编译。同时涵盖了通用Makefile的特性、解析和设计思路,适合开发者理解和实践。
摘要由CSDN通过智能技术生成

主要参考百问网科技:http://wiki.100ask.org/

1 Makefile 规则与示例

参考:
doc/gunmake.htm

1.1 为什么需要 Makefile

1.1.1 高效地编译程序

参考 Visual Studio ,修改源文件或者头文件,只会重新编译 修改过的文件,就可以生成 APP

1.1.2 Makefile 其实挺简单

一个简单的 Makefile 文件包含一系列的“规则”,其样式如下:

目标(target)…: 依赖(prerequiries)…
	<tab>命令(command)

如果“依赖文件”比“目标文件”更加新,那么执行“命令”来重新生成“目标文件”。
命令被执行的 2 个条件:依赖文件比目标文件新,或是 目标文件还没生成。

1.2 Makefile 的 2 个函数

1.2.1 $(foreach var,list,text)
  • list 中的每一个元素,取出来赋给 var,然后把 var 改为 text 所描述的形式

  • 示例

    objs := a.o b.o
    dep_files := $(foreach f, $(objs), .$(f).d)    # 最终 dep_files := .a.o.d .b.o.d
    
1.2.2 $(wildcard pattern)
  • pattern 所列出的文件是否存在,把存在的文件都列出来

  • 示例

    src_files := $(wildcard *.c)    # 最终 src_files 中列出了当前目录下的所有.c 文件
    

1.3 一步一步完善 Makefile

源码:
src/test_Makefile

1.3.1 示例
  • 第 1 个 Makefile ,简单粗暴,效率低

    test : main.c sub.c sub.h
    	gcc -o test main.c sub.c
    
  • 第 2 个 Makefile ,效率高,相似规则太多太啰嗦,不支持检测头文件

    test : main.o sub.o
        gcc -o test main.o sub.o
    
    main.o : main.c
        gcc -c -o main.o main.c
    
    sub.o : sub.c
        gcc -c -o sub.o sub.c
    
    clean:
        rm *.o test -f
    
  • 第 3 个 Makefile ,效率高,精炼,不支持检测头文件

    test : main.o sub.o
    	gcc -o test main.o sub.o
    
    %.o : %.c
        gcc -c -o $@ $<
    
    clean:
        rm *.o test -f
    
  • 第 4 个 Makefile ,效率高,精炼,支持检测头文件,但是需要手工添加头文件规则

    test : main.o sub.o
        gcc -o test main.o sub.o
    
    %.o : %.c
        gcc -c -o $@ $<
    
    sub.o : sub.h
    
    clean:
        rm *.o test -f
    
  • 第 5 个 Makefile ,效率高,精炼,支持自动检测头文件

    objs := main.o sub.o
    
    test : $(objs)
        gcc -o test $^
    
    # 需要判断是否存在依赖文件
    # .main.o.d .sub.o.d
    dep_files := $(foreach f, $(objs), .$(f).d)
    dep_files := $(wildcard $(dep_files))
    
    # 把依赖文件包含进来
    ifneq ($(dep_files),)
        include $(dep_files)
    endif
    
    %.o : %.c
        gcc -Wp,-MD,.$@.d -c -o $@ $<
    
    clean:
        rm *.o test -f
    
    distclean:
        rm $(dep_files) *.o test -f
    
1.3.2 规则
  • $@ :表示目标文件
  • $^ :表示所有的依赖文件
  • $< :表示第一个依赖文件

2 通用 Makefile 的使用

源码:src/general_Makefile

2.1 简介

百问网 参考 Linux 内核的 Makefile 编写了一个通用的 Makefile,它可以用来编译应用程序。

2.2 特点

  • 支持多个目录、多层目录、多个文件;

  • 支持给所有文件设置编译选项;

  • 支持给某个目录设置编译选项;

  • 支持给某个文件单独设置编译选项;

  • 简单、好用。

3 通用 Makefile 的解析

3.1 零星知识点

3.1.1 make 命令的使用

执行 make 命令时,它会去当前目录下查找名为 Makefile 的文件,并根据它的指示去执行操作,生成第一个目标。

  • 可以使用 -f 项指定文件,不再使用名为 Makefile 的文件

    $ make -f Makefile.build
    
  • 可以使用 -C 选项指定目录,切换到其他目录里去

    $ make -C a/ -f Makefile.build
    
  • 可以指定目标,不再默认生成 第一个目标

    $ make -C a/ -f Makefile.build other_target
    
3.1.2 即时变量,延时变量
# 变量的定义语法形式如下
A = xxx      # 延时变量,在使用时才展开、才确定
B ?= xxx     # 延时变量,只有第一次定义时赋值才成功;如果曾定义过,此赋值无效
C := xxx     # 立即变量
D += yyy     # 如果 D 在前面是延时变量,那么现在它还是延时变量
             # 如果 D 在前面是立即变量,那么现在它还是立即变量

GNU make 中对变量的赋值有两种方式:延时变量、立即变量。

3.1.3 变量的导出
  • 在编译程序时,会不断地使用 make -C dir 切换到其他目录,执行其他目录里的 Makefile
  • 如果想让某个变量的值在所有目录中都可见,要把它 export 出来。
  • 比如 CC = $(CROSS_COMPILE)gcc
    • 这个 CC 变量表示编译器,在整个过程中都是一样的。
    • 定义它之后,要使用 export CC 把它导出来。
3.1.4 Makefile 中可以使用 shell 命令
TOPDIR := $(shell pwd)      # 这是个立即变量, TOPDIR 等于 shell 命令 pwd 的结果
3.1.5 Makefile 中怎么放置第 1 个目标
  • 执行 make 命令时如果不指定目标,那么它默认是去生成第 1 个目标。所以 第 1 个目标 ,位置很重要。

  • 有时候不太方便把第 1 个目标完整地放在文件前面,这时可以在文件的前面直接放置目标,在后面再完善它的依赖与命令。

    First_target:  # 这句话放在前面
    ......         # 其他代码,比如 include 其他文件得到后面的 xxx 变量
    First_target : $(xxx) $(yyy) # 在文件的后面再来完善
        command
    
3.1.6 假想目标
  • Makefile 中有这样的目标

    clean:
        rm -f $(shell find -name "*.o")
        rm -f $(TARGET)
    

    如果当前目录下有名为 clean 的文件,执行 make clean 时它就不会执行那些删除命令。这时需要把 clean 这个目标,设置为 假想目标 ,这样可以确保执行 make clean 时,删除命令肯定可以得到执行。

  • 使用下面的语句把 clean 设置为假想目标

    .PHONY : clean
    

3.2 常用的函数

3.2.1 $(foreach var,list,text)
  • list 中的每一个元素,取出来赋给 var,然后把 var 改为 text 所描述的形式。

  • 示例

    objs := a.o b.o
    dep_files := $(foreach f, $(objs), .$(f).d)    # 最终 dep_files := .a.o.d .b.o.d
    
3.2.2 $(wildcard pattern)
  • pattern 所列出的文件是否存在,把存在的文件都列出来。

  • 示例

    src_files := $(wildcard *.c)    # 最终 src_files 中列出了当前目录下的所有.c 文件
    
3.2.3 $(filter pattern…,text)
  • text 中符合 pattern 格式的内容,filter 出来,留下来。

  • 示例

    obj-y := a.o b.o c/ d/
    DIR := $(filter %/, $(obj-y))     # 结果为:c/ d/
    
3.2.4 $(filter-out pattern…,text)
  • text 中符合 pattern 格式的内容,filter-out 出来、扔掉。

  • 示例

    obj-y := a.o b.o c/ d/
    DIR := $(filter-out %/, $(obj-y))    # 结果为: a.o b.o
    
3.2.5 $(patsubst pattern,replacement,text)
  • 寻找 text 中符合格式 pattern 的字,用 replacement 替换它们。 patternreplacement 中可以使用通配符。

  • 示例

    subdir-y := c/ d/
    subdir-y := $(patsubst %/, %, $(subdir-y))     # 结果为: c d
    

3.3 通用 Makefile 的设计思想

3.3.1 在 Makefile 中确定要编译的文件、目录,例如
obj-y += main.o
obj-y += a/

Makefile 文件总是被 Makefile.build 包含的。

3.3.2 在 Makefile.build 中设置编译规则,有 3 条编译规则
  • 进入子目录编译

    $(subdir-y):
        make -C $@ -f $(TOPDIR)/Makefile.build
    
  • 编译当前目录中的文件

    %.o : %.c
        $(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -Wp,-MD,$(dep_file) -c -o $@ $<
    
  • 当前目录下的 .o 和子目录下的 built-in.o 要打包起来

    built-in.o : $(cur_objs) $(subdir_objs)
        $(LD) -r -o $@ $^
    
3.3.3 顶层 Makefile 中把顶层目录的 built-in.o 链接成 APP
$(TARGET) : built-in.o
    $(CC) $(LDFLAGS) -o $(TARGET) built-in.o

4 总结

4.1 添加宏定义

​ 添加宏定义是利用 gcc 的语法规则,一般是为了再编译过程中进行功能的定制区分。

4.1.1 Makefile 应该怎样修改
TEST = $(test)
ifeq ($(TEST),1)
	CFLAGS += -DTEST_DEF
#$(info TEST = $(TEST))
endif
4.1.2 C 程序里怎么使用
#include <stdio.h>


void fun1()
{
	printf("this is fun1\n");
}

void fun2()
{
	printf("this is fun2\n");
}

int main(int argc, char *argv[])
{
	fun1();
#ifdef TEST_DEF
	fun2();
#endif
	return 0;
}
4.1.3 测试
  • 不加参数

    # 一定要 clean
    $ make clean
    
    # 编译
    $ make
    
    # 执行
    $ ./test
    this is fun1
    
  • 加参数

    # 一定要 clean
    $ make clean
    
    # 编译,有参数
    $ make test=1
    
    # 执行
    $ ./test
    this is fun1
    this is fun2
    
摘自http://www.linuxsir.org/main/doc/gnumake/GNUmake_v3.80-zh_CN_html 由徐海兵老師翻译整理。 本文比较完整的讲述GNU make工具,涵盖GNU make的用法、语法。同时重点讨论如何为一个工程编写Makefile。作为一个Linux程序员,make工具的使用以及编写Makefile是必需的 目 录 第一章:概述 1.1 概述 1.2 准备知识 第二章:GNU make 介绍 2.1 Makefile简介 2.2 Makefile规则介绍 2.3 简单的示例 2.4 make如何工作 2.5 指定变量 2.6 自动推导规则 2.7 另类风格的makefile 2.8 清除工作目录过程文件 第三章:Makefile 总述 3.1 Makefile的内容 3.2 makefile文件的命名 3.3 包含其它makefile文件 3.4 变量 MAKEFILES 3.5 变量 MAKEFILE_LIST 3.6 其他特殊变量 3.7 makefile文件的重建 3.8 重载另外一个makefile 3.9 make如何解析makefile文件 3.9.1 变量取值 3.9.2 条件语句 3.9.3 规则的定义 3.10 总结 第四章:Makefile的规则 4.1 一个例子 4.2 规则语法 4.3 依赖的类型 4.4 文件名使用通配符 4.4.1 统配符使用举例 4.4.2 通配符存在的缺陷 4.4.3 函数wildcard 4.5 目录搜寻 4.5.1 一般搜索(变量VPATH) 4.5.2 选择性搜索(关键字vpath) 4.5.3 目录搜索的机制 4.5.4 命令行和搜索目录 4.5.5 隐含规则和搜索目录 4.5.6 库文件和搜索目录 4.6 Makefile伪目标 4.7 强制目标(没有命令或依赖的规则) 4.8 空目标文件 4.9 Makefile的特殊目标 4.10 多目标 4.11 多规则目标 4.12 静态模式 4.12.1 静态模式规则的语法 4.12.2 静态模式和隐含规则 4.13 双冒号规则 4.14 自动产生依赖 第五章:规则的命令 5.1 命令回显 5.2 命令的执行 5.3 并发执行命令 5.4 命令执行的错误 5.5 中断make的执行 5.6 make的递归执行 5.6.1 变量MAKE 5.6.2 变量和递归 5.6.3 命令行选项和递归 5.6.4 -w选项 5.7 定义命令包 5.8 空命令 第六章:Makefile中的变量 6.1 变量的引用 6.2 两种变量定义(赋值 ) 6.2.1 递归展开式变量 6.2.2 直接展开式变量 6.2.3 如何定义一个空格 6.2.4 “?=”操作符 6.3 变量的高级用法 6.3.1 变量的替换引用 6.3.2 变量的套嵌引用 6.4 变量取值 6.5 如何设置变量 6.6 追加变量值 6.7 override 指示符 6.8 多行定义 6.9 系统环境变量 6.10 目标指定变量 6.11 模式指定变量 第七章:Makefile的条件执行 7.1 一个例子 7.2 条件判断的基本语法 7.2.1 关键字“ifeq” 7.2.2 关键字“ifneq” 7.2.3 关键字“ifdef” 7.2.4 关键字“ifndef” 7.3 标记测试的条件语句 第八章:make的内嵌函数 8.1 函数的调用语法 8.2 文本处理函数 8.2.1 $(subst FROM,TO,TEXT) 8.2.2 $(patsubst PATTERN,REPLACEMENT,TEXT) 8.2.3 $(strip STRINT) 8.2.4 $(findstring FIND,IN) 8.2.5 $(filter PATTERN…,TEXT) 8.2.6 $(filter-out PATTERN...,TEXT) 8.2.7 $(sort LIST) 8.2.8 $(word N,TEXT) 8.2.9 $(wordlist S,E,TEXT) 8.2.10 $(words TEXT) 8.2.11 $(firstword NAMES…) 8.3 文件名处理函数 8.3.1 $(dir NAMES…) 8.3.2 $(notdir NAMES…) 8.3.3 $(suffix NAMES…) 8.3.4 $(basename NAMES…) 8.3.5 $(addsuffix SUFFIX,NAMES…) 8.3.6 $(addprefix PREFIX,NAMES…) 8.3.7 $(join LIST1,LIST2) 8.3.8 $(wildcard PATTERN) 8.4 foreach 函数 8.5 if 函数 8.6 call函数 8.7 value函数 8.8 eval函数 8.9 origin函数 8.10 shell函数 8.11 make的控制函数 8.11.1 $(error TEXT…) 8.11.2 $(warning TEXT…) 第九章:执行make 9.1 指定makefile文件 9.2 指定终极目标 9.3 替代命令的执行 9.4 防止特定文件重建 9.5 替换变量定义 9.6 使用make进行编译测试 9.7 make的命令行选项 第十章:make的隐含规则 10.1 隐含规则的使用 10.2 make的隐含规则一览 10.3 隐含变量 10.3.1 代表命令的变量 10.3.2 命令参数的变量 10.4 make隐含规则链 10.5 模式规则 10.5.1 模式规则介绍 10.5.2 模式规则示例 10.5.3 自动化变量 10.5.4 模式的匹配 10.5.5 万用规则 10.5.6 重建内嵌隐含规则 10.6 缺省规则 10.7 后缀规则 10.8 隐含规则搜索算法 第十一章:使用make更新静态库文件 11.1 库成员作为目标 11.2 静态库的更新 11.2.1 更新静态库的符号索引表 11.3 make静态库的注意事项 11.4 静态库的后缀规则 第十二章 : GNU make的特点 12.1 源自System v的特点 12.2 源自其他版本的特点 12.3 GNU make自身的特点 第十三章 和其它版本的兼容 第十四章 Makefile的约定 14.1 基本的约定 14.2 规则命令行的约定 14.3 代表命令变量 14.4 安装目录变量 14.5 Makefile的标准目标名 14.6 安装命令分类 第十五章 make的常见错误信息   附录:关键字索引 1. GNU make可识别的指示 符 2. GNU make函数 3. GNU make的自动化变量 4. GNU make环境变量 后序
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值