makefile详解

本文深入讲解Makefile的编写技巧,包括目标、依赖和命令的基本规则,以及如何利用Makefile的更新机制实现高效编译。文章还介绍了Makefile中的变量使用,如自定义变量、自动变量和系统维护的变量,并展示了如何通过函数简化大型项目的源文件管理。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、makefile的编写

        1)makefile的命名

        2)规则中的三要素:目标,依赖,命令

    固定格式

        目标:依赖条件

        Tab键  命令(gcc)

eg:   

 app:main.c  add.c  sub.c  mul.c

          gcc main.c add.c sub.c mul.c -o app

          如果.c不在同一级目录,须指定目录

上面方法的缺点就是:每一次make都会编译所有的文件

基于以上缺点改进的新规则

      app:main.o add.o sub.o mul.o

            gcc main.o add.o sub.o mul.o -o app

      main.o:main.c
            gcc main.c -c main.o

      add.o:add.c
            gcc add.c -c add.o

      mul.o:mul.c
            gcc mul.c -c mul.o

优点:当其中某个.c文件改变只需要编译某个.c文件,而不需要全部编译

Makefile 的更新机制,通过对比目标文件和依赖文件的更新时间,如果目标文件的更新时间早于依赖文件则开始编译;

以上例子为例,如果makefile中有多个目标,那么第一条中的目标就是我们的终极目标

Makefile中的变量:包括自定义变量  自动变量 自己维护的变量

自定义变量

eg   obj = main.o add.o sub.o mul.o

       $(obj)提取变量里面的值

替换: 3,4 s/app/g/$(target)

自动变量:

 $<  指的是所有依赖文件中的第一个依赖

 $^  依赖文件中所有的依赖文件

 $@  规则中的目标

系统维护的变量:

CC = cc = gcc
CPPFLAGS = -I  //指定头文件
CFLAGS = -Wall -g -c //编译时使用的参数
LDFLAGS = -L -l//链接库使用的选项-L -i

通过学习以上变量后,重新编写makefile

.PHONY clean
obj = main.o mul.o add.o mul.o
target = myapp
CC = gcc
CFLAGS = -g -Wall -c
$(target):$(obj)
    $(CC) $(CFLAGS) $(obj) -o $@ 
%.o:%.c
    $(CC) -c %.c
clean:
    rm $(target) %(obj) -rf

终极版本

在一个项目中我们可能会有很多的源文件,如果一一的列在makefile 文件中会使得Makefile文件变得冗长,于是为了解决这个问题,我们引入了makefile函数

$(wildcard    参数(路径/*.c)
$(patsubst     路径/%.c , 路径/%.o ,$(src))

通过$符号获取函数返回值 

其中第二条命令用%而不用*原因是为了模式匹配,即一个.c文件生成一个对应的.o文件。

./PHONY clean//声明clean 为一个伪目标,也就是不会再把本地磁盘做更新比较。

clean:

        rm $(obj)       $(target)         -f (强制执行)

-mkdir  /aa //在mkdir前面加了-  表示如果当前命令执行失败,就忽略本次命令,继续执行后面的命令

 

七、静态模式 静态模式可以更加容易地定义多目标的规则,可以让我们的规则变得更加的有弹性和灵活。我们还是先来看一下语法: ;: ;: ; ; ... targets定义了一系列的目标文件,可以有通配符。是目标的一个集合。 target-parrtern是指明了targets的模式,也就是的目标集模式。 prereq-parrterns是目标的依赖模式,它对target-parrtern形成的模式再进行一次依赖目标的定义。 这样描述这三个东西,可能还是没有说清楚,还是举个例子来说明一下吧。如果我们的;定义成“%.o”,意 思是我们的;集合中都是以“.o”结尾的,而如果我们的;定义成“%. c”,意思是对;所形成的目标集进行二次定义,其计算方法是,取;模式中的“%”(也就是去掉了[.o]这个结尾),并为其加上[.c]这个结尾,形成的新集合。 所以,我们的“目标模式”或是“依赖模式”中都应该有“%”这个字符,如果你的文件名中有“%”那么你可以使用反斜杠“\”进行转义,来标明真实的“%”字符。 看一个例子: 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.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 $< $(filter %.o,$(files))表示调用Makefile的filter函数,过滤“$filter”集,只要其中模式为“%.o”的内容。其的它内容,我就不用多说了吧。这个例字展示了Makefile中更大的弹性。 八、自动生成依赖性 在Makefile中,我们的依赖关系可能会需要包含一系列的头文件,比如,如果我们的main.c中有一句“#include "defs.h"”,那么我们的依赖关系应该是: main.o : main.c defs.h
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值