我的生成文件:
compiler=g++
cflags=-g -Wall -I.
src=$(shell find . -name *.cc) #find all .cc files, with path name
srcBaseName=$(shell basename -a $(src)) # extract base names by stripping off the path
header=$(shell find . -name *.h) # all header files
obj=$(patsubst %.cc, %.o, $(srcBaseName)) # Problematic line
bin=bin/myProgram
all: $(bin)
$(bin): $(obj)
$(compiler) $^ -o $@
%.o: %.cc
$(compiler) $(cflags) -c $^ -o $@
clean:
rm $(obj) $(bin)
导致以下错误:
make: *** No rule to make target 'SomeObjectFile.o', needed by 'bin/myProgram'. Stop.
有问题的行:
obj=$(patsubst %.cc, %.o, $(srcBaseName))
如果我将 $(srcBaseName) 更改为 $(src),那么一切都会正常构建。但在那种情况下,.o 文件将与 .cc 文件分散在相应的文件夹中,这是我不希望的。
我想要一个专门的 (obj/) 文件夹来存储所有的 .o 文件。
我该怎么做?
第一次尝试:
obj=$(patsubst %.cc, %.o, obj/$(srcBaseName))
第二次尝试:
obj=$(patsubst %.cc, %.o, obj\/$(srcBaseName))
为什么它们不起作用?
/************************编辑于 2015 年 2 月 16 日********************* ***/
根据答案中的建议,我将 Makefile 更新为:
compiler=g++
# source compilation flags
cflag=-g -Wall -std=gnu++0x -I./header/
# source link flags
lflag=
# lib used by proj
lib=
tflag=-g -Wall -std=gnu++0x
# test link flags
tlflag=
# test libs
testLib=lib/libgtest.a
# source code
src=$(shell find "./src" -name "*.cc")
srcBaseName=$(shell basename -a $(src))
obj=$(addprefix obj/, $(patsubst %.cc, %.o, $(srcBaseName)))
vpath %.cc $(dir $(src))
# header files
header=$(shell find "./header" -name "*.h")
# test files
testSrc=$(shell find "./test" -name "*.cc")
testSrcBase=$(shell basename -a $(testSrc))
testObj=$(addprefix obj/, $(patsubst %.cc, %.o, $(testSrcBase)))
vpath %.cc $(dir $(testSrc))
# binary files
bin=bin/Driver
testBin=bin/Test
all: prog test
prog: $(bin)
$(bin): $(obj)
$(compiler) $(lflag) $^ $(lib) -o $@
#$(obj): $(src) $(header)
obj/%.o: %.cc $(header)
$(compiler) $(cflag) -c $< -o $@
test: $(testBin)
$(testBin): $(testObj)
$(compiler) $(tlflag) $^ $(testLib) -o $@
obj/%.o: %.cc
$(compiler) $(tflag) -c $< -o $@
clean:
rm $(obj) $(bin) $(testObj) $(testBin)
这是制作背后的意图:
制作程序:
make应该找到./src目录下的所有源文件(.cc),并在./obj目录下生成一个同名的.o文件,对子目录的层级不敏感,可以随意添加无需更新 Makefile 的新 cc 文件。每个 .o 文件都依赖于相应的(只是同名的那个,而不是全部).cc 文件和所有头文件(make 不会在不解析文件的情况下自动知道 cc 文件包含哪些头文件;如果您有实现此目的的巧妙方法,请告诉我!)。例如,./src/subdirectory1/sample1.cc 应该产生 ./obj/sample1.o 和 ./obj/sample1.o 取决于 ./src/subdirectory1/sample1.cc + ./header/sample1.h + 。/header/sample2.h + ...
进行测试:
它应该与 ./test 文件夹中的测试源文件做类似的事情,除了不涉及标题。如果此详细信息有帮助,我正在使用 Google Test。
但是,我的 Makefile 并没有按照预期的方式工作,因为它存在以下问题:
1,如果我运行 make test
, 食谱 $(compiler) $(tflag) -c $< -o $@
未执行(tflag 表示'测试编译标志',没有-I./header/部分;cflag 表示'源代码编译标志',它有-I./header/部分)。相反,伪编的食谱 $(compiler) $(cflag) -c $< -o $@
被执行。此观察结果来自出现“-I./header/”的输出。我想这是因为 phony prog 中的 cflag 模式规则覆盖了 phony test 中的 tflag 规则?我依稀记得make选择了最匹配的模式规则-两者本质上是相同的(我的意图是在执行该假冒时应执行特定假冒下的模式规则,这似乎不可行?),所以 make 总是会选择第一个。通过交换 Makefile 中两个模式规则的顺序来验证这个结论,这次 tflag 规则总是被选中。 因此,一个自然而然的问题是,当手机被执行时,我如何在特定的假手机下执行模式规则?
2,假设第1点我想做的事情不可行,我开始考虑替代方案。我可以做类似的事情吗:#$(obj): $(src) $(header)
这样我就可以摆脱模式规则来解决 make 选择模式规则的方式。但是,这显然是不正确的,$(obj) 中的每个 .o 文件都依赖于所有 src 文件和所有头文件。 这是一个正确的方向吗?
非常感谢,期待您的来信。
3 个关键问题已以粗体和斜体突出显示。
最佳答案
你的问题是这一行:
%.o: %.cc
该行告诉 make 创建 some/path/file.o 你将使用 some/path/file.cc。
如果您希望所有 .o 文件都在一个目录中,但仍希望将源文件放在不同的目录中,则每个源目录都需要一个这样的规则。或者,您可以将所有目录添加到 VPATH 变量,例如:
VPATH=$(dir $(src))
或者可能更好:
VPATH=$(dir $(SRC))
在 Makefile 中为变量使用大写字母是避免将它们与函数名称混淆的好方法。
关于c++ - Makefile:无法获得与 patsubst 一起使用的基名,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26445523/