0

我正在尝试使用GNU make(版本 3.81)构建一个小型 C++ 项目,但我必须调用make两次,因为第一次运行失败。这是我的项目目录:

project
    makefile
    include
        lexer.hpp   
    src
        main.cpp
        lexer.l

以下是我的makefile:

CC = g++
CPPFLAGS = -I include
VPATH = include src

OBJECTS = main.o lexer.o

test: $(OBJECTS)
$(CC) $(CPPFLAGS) -lfl -o $@ $^

main.o: lexer.hpp main.cpp
    $(CC) -c $(CPPFLAGS) $^

lexer.o: lexer.cpp
    $(CC) -c $(CPPFLAGS) $^

lexer.cpp: lexer.l
    flex -t $^ > src/lexer.cpp

.PHONY: clean

clean:
    rm -fR $(OBJECTS) src/lexer.cpp test

我第一次运行make我得到以下输出,其中make抱怨找不到 lexer.cpp 文件。但我不明白为什么make不在 VPATH 中声明的文件夹中查找。

g++ -c -I include include/lexer.hpp src/main.cpp
flex -t src/lexer.l > src/lexer.cpp
g++ -c -I include lexer.cpp
g++: error: lexer.cpp: No such file or directory
g++: fatal error: no input files
compilation terminated.
make: *** [lexer.o] Error 1

但是,如果我再次调用make然后找到 lexer.cpp 并且编译工作。

g++ -c -I include src/lexer.cpp
g++ -I include -lfl -o test main.o lexer.o

为什么?

PS我为糟糕的英语道歉。

4

1 回答 1

2

这个规则是错误的:

lexer.cpp: lexer.l
        flex -t $^ > src/lexer.cpp

该规则告诉 make 它将构建一个文件lexer.cpp,这就是 make 期望它做的事情,并且在规则完成后,make 认为该文件已准备好运行,并且当其他目标依赖它时它将使用该文件名。但是该规则真正起作用的是 build src/lexer.cpp

要正确编写此规则,您需要将其编写为:

src/lexer.cpp: lexer.l
        flex -t $^ > $@

(您编写的每个 make 规则都应始终$@准确地更新文件)。

但是,通常 VPATH 不适合查找生成的文件(目标文件等:由 make 生成的任何文件)。它仅对查找源文件有用(文件 make 不会自行构建)。

于 2013-07-19T15:09:19.400 回答