在GNU Makefile中(在Ubuntu Linux系统上),如何删除列表中每个文件名的文件名后缀,使文件名在第一个点被截断?
假设我有
NAMES = file1.a.b.c file2.x.y.z file3.b file4.foo
其中名称可能来自使用$(wildcard file*)
的文件名扩展扩展。
我想结束
NEWNAMES = file1 file2 file3 file4
甚至可能
NEWNAMES = file1.quux file2.quux file3.quux file4.quux
(不一定按此顺序)
$(basename ...)
函数仅从名称中删除最后一个后缀。
在支持数组的shell中,我会使用像"${names[@]%%.*}"
或"${names[@]/%.*/.quux}"
这样的东西。
想要这样做的原因是我们正在使用生物信息学环境,其中文件具有已知的文件名前缀,但它们的后缀可以是.fa
,.fas
,.fasta
(等)的任意组合,这些后缀的可能性加倍up(如在.fa.fa
中)并且最后还有一个文件压缩器后缀,例如.gz
。
无论初始文件名后缀有多复杂,我们都希望将文件名转换为Makefile中的规范化prefix.suffix
文件名。
如果你想在不使用shell的情况下使用make函数,你可以使用这样的东西:
NAMES := $(foreach F,$(wildcard file*),$(firstword $(subst ., ,$F)))
我不确定这里有什么平台(我们是否有SUSv3兼容/ bin / sh)以及对名称的任何要求。
如果名称不能包含空格是可以接受的,我认为这很简单
NEWNAMES = $(shell for p in $(NAMES); do echo $${p%%.*};done)
足够接近,易于理解。
如果您想要更灵活的方法,可以使用gmtt,一个GNUmake库:
include gmtt-master/gmtt-master/gmtt.mk
NAMES = file1.a.b.c file2.x.y.z file3.b file4.foo
$(info $(foreach fn,$(NAMES),$(call glob-match,$(fn),*.*)$(newline)))
NEWNAMES = file1.middle.quux file2.middle.quux file3.middle.quux file4.more.middle.quux
$(info $(foreach fn,$(NEWNAMES),$(call glob-match,$(fn),*.*.quux)$(newline)))
输出:
$ make
file1 . a.b.c
file2 . x.y.z
file3 . b
file4 . foo
file1 . middle .quux
file2 . middle .quux
file3 . middle .quux
file4 . more.middle .quux
make: *** Keine Ziele. Schluss.
glob-matcher返回一个可以进一步处理的分隔字符串列表。