PHONY在Makefile中是什么意思?我已经经历过了,但这太复杂了。

有人能简单地向我解释一下吗?


当前回答

我经常用它们来告诉默认目标不要开火。

superclean: clean andsomethingelse

blah: superclean

clean:
   @echo clean

%:
   @echo catcher $@

.PHONY: superclean

如果没有PHONY,使超级清洁将发射干净,以及其他一些东西,并捕捉超级清洁;但是有了PHONY,make superclean就不会解雇捕手superclean了。

我们不必担心会告诉对方,明确的目标是PHONY,因为它并不完全是假的。虽然它从未生成干净的文件,但它有命令可以启动,因此make会认为它是最终目标。

然而,超精简目标确实是假的,因此make将尝试将其与为超精简目标提供dep的任何其他内容叠加起来-这包括其他超精简目标和%目标。

请注意,我们根本没有说任何关于或其他什么的话,所以他们很明显会去找捕手。

输出如下所示:

$ make clean
clean

$ make superclean
clean
catcher andsomethingelse

$ make blah 
clean
catcher andsomethingelse
catcher blah

其他回答

.PHONY: install

表示单词“install”不代表此文件中的文件名生成文件;表示Makefile与名为“install”的文件无关在同一目录中。

特殊目标.PHONY:允许声明虚假目标,这样make就不会将它们作为实际文件名进行检查:即使这样的文件仍然存在,它也会一直工作。

你可以在Makefile中放入几个。PHONY:

.PHONY: all

all : prog1 prog2

...

.PHONY: clean distclean

clean :
    ...
distclean :
    ...

还有另一种方法可以声明虚假目标:只需::无需先决条件:

all :: prog1 prog2

...

clean ::
    ...
distclean ::
    ...

::还有其他特殊的含义,请参见这里,但在没有前提条件的情况下,它总是执行食谱,即使目标已经存在,因此充当了一个虚假的目标。

最好的解释是GNU make手册本身:4.6 Phony Targets部分。

.PHONY是make的特殊内置目标名称之一。还有其他你可能感兴趣的目标,所以值得浏览一下这些参考文献。

当该考虑.PHONY目标时,make将运行其配方无条件地,无论具有该名称的文件是否存在或它的最后修改时间是什么。

你也可能对make的标准目标感兴趣,比如all和clean。

“.PHONY”还有一个重要的棘手之处——当一个物理目标依赖于另一个物理对象的虚假目标时:

目标1->PHONY_FORWARDER1->PHONY-FORWARDER2->TARGET2

您可以简单地预期,如果您更新了TARGET2,那么TARGET1应该被认为是过时的,因此TARGET1应该重新构建。它确实是这样工作的。

棘手的部分是,当TARGET2与TARGET1不过时时——在这种情况下,您应该预计TARGET1不应该重建。

这令人惊讶地不起作用,因为:无论如何都运行了假目标(就像假目标通常所做的那样),这意味着假目标被认为是更新的。正因为如此,TARGET1被认为是过时的,而不是虚假的目标。

考虑:

all: fileall

fileall: file2 filefwd
    echo file2 file1 >fileall


file2: file2.src
    echo file2.src >file2

file1: file1.src
    echo file1.src >file1
    echo file1.src >>file1

.PHONY: filefwd
.PHONY: filefwd2

filefwd: filefwd2

filefwd2: file1
    @echo "Produced target file1"


prepare:
    echo "Some text 1" >> file1.src
    echo "Some text 2" >> file2.src

你可以玩这个:

首先做“make prepare”以准备“源文件”通过触摸特定文件来查看它们的更新

您可以看到,fileall通过一个虚假的目标间接依赖于file1,但由于这种依赖性,它总是被重新构建。如果您将fileall中的依赖项从filefwd更改为file,那么现在fileall不会每次都重新生成,而是只有在任何依赖目标作为文件过时时才会重新生成。

假设您有安装目标,这在makefile中很常见。如果不使用.PHONY,并且名为install的文件与Makefile位于同一目录中,那么makeinstall将不会执行任何操作。这是因为Make将规则解释为“执行这样和那样的配方以创建名为install的文件”。由于文件已经存在,并且其依赖关系没有更改,因此将不会执行任何操作。

但是,如果您将安装目标设置为PHONY,它会告诉make工具该目标是虚构的,make不应该期望它创建实际的文件。因此,它不会检查安装文件是否存在,这意味着:a)如果文件确实存在,它的行为不会改变;b)不会调用额外的stat()。

通常,Makefile中的所有目标,如果没有生成与目标名称相同的输出文件,则应为PHONY。这通常包括all、install、clean、distclean等。