makefile-伪目标语法与实例分析

本文深入探讨了makefile中的伪目标概念及其作用。通过一系列编程实验,展示了如何利用伪目标来确保命令的执行,无论目标依赖是否更新。实验涵盖了从基本的文件处理到使用.PHONY关键字声明伪目标,以及绕开.PHONY的其他方法。通过这些实验,读者可以掌握如何使用伪目标来模拟C语言中的函数调用,并理解其在清理目标和强制执行命令中的应用。

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

makefile中的目标的含义:

默认情况下,
  • make认为目标对应着一个文件
  • make比较目标文件和依赖文件的新旧关系,决定是否执行命令
  • make以文件处理作为第一优先级

编程实验1:(make默认认为目标对应着一个文件)

当前目录结构如下:
示例
func.c源文件如下:

void fun()
{
    printf("void fun():hello makefile \n");
}

main.c源文件如下:

extern void fun();

int main()
{
    fun();

    return 0;
}

makefile源文件如下:

hello.out:main.o func.o
        gcc -o hello.out main.o func.o

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

func.o:func.c
        gcc -o func.o -c func.c

clean:
        rm *.o hello.out

命令行执行结果如下:

分析: 执行make clean命令会将当前目录下的所有.o文件和hello.out一起删除;而当再次执行make clean的时候由于当前目录下已经没有.o文件和hello.out文件,所以会出现提示信息无法移除文件

编程实验2:(make以文件处理作为第一优先级)

当前目录结构如下:
示例
main.c func.c makefile文件内容均与编程实验1保持一致,命令行执行结果如下:

分析: 由于当前目录下存在文件clean,故当执行make clean的命令时,make以文件处理作为第一优先级,而当前文件已经是最新的,故不会再执行make clean命令。
出现此问题的解决方法为增加伪目标,,具体分析请看编程实验3。

编程实验3:(伪目标的引入,保证与目标的依赖无关,目标中的命令总是被执行)

当前目录结构如下:
示例
main.c func.c 文件内容均与编程实验1保持一致,makefile文件做如下修改(在clean目标之前增加代码:.PHONY:clean):

## makefile
hello.out :main.o func.o
    gcc -o hello.out main.o func.o

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

func.o:func.c
    gcc -o func.o -c func.c

.PHONY:clean

clean:
    rm *.o hello.out

命令行执行结果如下:

示例

伪目标知识:

  • 通过.PHONY关键字声明一个伪目标
  • 伪目标不对应任何实际的文件
  • 不管伪目标的依赖是否更新,命令总是执行
  • 伪目标的语法:先声明,后使用
    本质:
    伪目标是make中特殊目标.PHONY的依赖

编程实验4(使用伪目标模拟C语言中的函数调用)

当前目录结构如下:
示例
main.c func.c 文件内容均与编程实验1保持一致,makefile文件做如下修改(在clean目标之前增加代码:.PHONY:clean):

## makefile
hello.out :main.o func.o
        gcc -o hello.out main.o func.o

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

func.o:func.c
        gcc -o func.o -c func.c

.PHONY:rebuild clean all

rebuild:clean all

all:hello.out

clean:
    rm *.o hello.out
  • 原理: 当一个目标的依赖包含伪目标时,伪目标所定义的命令总是会被执行
    命令行执行结果如下:

示例
分析: 由于reuild、clean、all被.PHONY定义为伪目标;故每次执行rebuild、all、clean目标的时候与目标对应的命令都会被执行;所以当执行make rebuild的时候,实际上是先执行clean命令(移除当前目录下所有的.o文件和hello.out文件),然后执行all命令(执行hello.out命令)

编程实验5(绕开.PHONY关键字定义伪目标 )

  • 原理: 如果一个规则没有命令或者依赖,并且它的目标不是一个存在的文件名;在执行此规则时,目标总是会被认为是最新的
    当前目录结构如下:
    示例
    main.c func.c 文件内容均与编程实验1保持一致,makefile文件做如下修改(将clean目标依赖于FORCE,而FORCE定义为空目标):
##makefile
hello.out :main.o func.o
    gcc -o hello.out main.o func.o

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

func.o:func.c
    gcc -o func.o -c func.c

clean:FORCE
    rm *.o hello.out

FORCE:

命令行执行结果如下:
示例
* 分析: 因为FORCE目标是最新的,所以每次执行make clean的时候都会执行clean的命令

小结:

  • 默认情况下,make认为目标对应着一个文件
  • .PHONY用于声明一个伪目标,伪目标不对应实际的文件
  • 伪目标的本质是make中特殊目标.PHONY的依赖
  • 使用伪目标可以模拟“函数调用”,方法请参考上文编程实验5
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值