c++ - Makefile:无法获得与 patsubst 一起使用的基名

标签 c++ makefile

我的生成文件:

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/

相关文章:

c++ - Boost 安装失败 : The system cannot find the path specified

c++ - 在 C++11 中声明 char 十六进制常量

linux - 当 gnu make phony 目标恰好与目录名称相同时的行为

c - 如何更正我的 Makefile,以便它构建我的所有源文件? (答案: check for hidden tabs)

生成文件: find recursively and delete auxiliary files

c++ - 如果一个引用一旦被初始化为一个对象,它就不能被改变,为什么会这样呢?

c++ - MinGW的Qt getline错误

c++ - 当我编写这个程序时,这些标志会被调用吗?

makefile - 防止两个 make 实例执行同一个目标(并行 make)

c++ - 合并每个 3 个元素的数组中彼此最接近的点