这是从这里开始的问题的延续。问题是有一个规则会从单个输入生成多个输出,并且该命令很耗时,因此我们希望避免重新计算。现在还有一个额外的转折,我们希望防止文件作为中间文件被删除,并且规则涉及通配符以允许参数。
建议的解决方案是我们设置以下规则:
file-a.out: program file.in
./program file.in file-a.out file-b.out file-c.out
file-b.out: file-a.out
@
file-c.out: file-b.out
@
然后,调用make file-c.out
创建了两者,我们避免了与switchmake
并行运行的问题。-j
到目前为止一切都很好。
问题如下。由于上述方案在DAG中建立了一个链,make
考虑不同;这些文件被视为中间文件,默认情况下,一旦准备好,它们就会被删除,因为它们是不必要file-a.out
的。file-b.out
file-c.out
此处某处提到了一种避免这种情况的方法,包括添加file-a.out
和file-b.out
作为目标的依赖项.SECONDARY
,以防止它们被删除。不幸的是,这并不能解决我的问题,因为我的规则使用通配符模式;具体来说,我的规则看起来更像这样:
file-a-%.out: program file.in
./program $* file.in file-a-$*.out file-b-$*.out file-c-$*.out
file-b-%.out: file-a-%.out
@
file-c-%.out: file-b-%.out
@
这样就可以传递包含在文件名中的参数,例如通过运行
make file-c-12.out
文档建议的解决方案make
是将这些作为隐式规则添加到 的依赖项列表中.PRECIOUS
,从而防止这些文件被删除。
该解决方案.PRECIOUS
有效,但它还可以防止在规则失败且文件不完整时删除这些文件。有没有其他方法可以使这项工作?
解决这个问题的一个技巧是定义一个.SECONDARY
没有先决条件的目标,即
.SECONDARY:
它通知make
所有文件都应该被视为次要文件,因此不会被删除,除非make
被中断或规则失败。不幸的是,这不允许选择带有通配符的规则子集以这种方式工作,所以我认为这只是一个 hack(尽管它很有用)。