为什么Makefile中的clean需要.PHONY

原因一:避免Makefile检查时间戳

前置知识:makefile在依赖文件没有改变时不会执行编译命令

#第一次执行,OK
[root@VM-16-14-centos ~]# make
g++  -E main.cc -o main.i
g++  -S main.i -o main.s
g++  -c main.s -o main.o
g++  main.o -o main

#第二次执行,源文件没有被修改,为了效率make不会执行编译
[root@VM-16-14-centos ~]# make
make: `main' is up to date.

当目录中存在名为clean的文件时,若未声明.PHONYmake clean会检查该文件的时间戳。由于clean目标通常没有依赖文件,Make会认为该目标已是最新状态,从而跳过命令执行。

# 未声明.PHONY时,若存在clean文件:
$ touch clean
$ make clean
make: 'clean' is up to date.  # 命令未执行

.PHONY声明后,无论文件是否存在,make clean都会执行其命令。例如:

.PHONY: clean
clean:
    rm -f *.o main

即使存在clean文件,也会强制删除目标文件。

情况未声明.PHONY声明.PHONY
目录中存在clean文件命令被跳过命令强制执行
clean文件命令正常执行命令正常执行

原因二:优化性能(次要)

在Makefile中,.PHONY目标通过跳过隐含规则搜索来提升性能,其原理和优化效果可分为以下四个层面:


场景未声明.PHONY声明.PHONY
目标文件已存在检查时间戳(约0.5ms)直接执行命令(0ms)
多级依赖链递归检查所有依赖仅执行命令不检查依赖
并行编译(make -j)可能因规则搜索阻塞线程减少线程竞争

  1. 必须声明为.PHONY的目标
    • cleaninstall等不生成文件的操作
    • 入口目标如all(避免与all文件冲突)
  2. 不应声明的情况
    • 实际需要生成文件的目标(如main.o),否则会重复编译,降低效率

例子:实现多版本编译的Makeflie

make -> 默认,按RELEASE模式编译
make DEBUG -> DEBUG选项,按DEBUG模式编译
make clean -> 清理选项,清理项目文件

FLAGS = 

ifeq ($(MAKECMDGOALS), DEBUG)
        FLAGS += -g
endif


main:main.o
        g++ $(FLAGS) main.o -o main
main.o:main.s
        g++ $(FLAGS) -c main.s -o main.o
main.s:main.i
        g++ $(FLAGS) -S main.i -o main.s
main.i:main.cc
        g++ $(FLAGS) -E main.cc -o main.i

.PHONY:clean
clean:
        rm -rf main.i main.s main.o main

DEBUG:main
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值