按照此链接中的建议,我已使用 #include 指令将 C++ 模板类拆分为两个文件 --> Templates spread across multiple files
对于这两个文件,我使用了标准命名 *.h 和 *.cpp。 在 makefile 中,我包含了 *.h 文件,因为据我了解,从编译器的角度来看,以这种方式分割的两个文件实际上都是一个。我设置了与编译常用 *.cpp 文件相同的编译选项。
但是,当我尝试通过 makefile 编译它时,出现以下错误:
Stos.o: file not recognized: File format not recognized
collect2: ld returned 1 exit status
make: *** [program] Error 1
我在某处读到这可能是由于我编译 *.h 文件所致。我不知道该怎么办。
感谢您的帮助。
编辑
我发布了 makefile 本身以使问题更加清楚。请原谅,这些名字不是英文的,但是在这种情况下这应该不是问题。我已经更新了错误消息,使其适合 makefile。
根据此处的建议,我已将模板文件的名称从上面提到的 *.h 和 *.cpp 更改为 *.hpp 和 *.tpp。
CC=g++
CFLAGS=-c -Wall -pedantic -W
COMP= $(CC) $(CFLAGS)
program: Graf.o wgraf.o wnazwe.o menu.o Stos.o TabDyn.o komunikaty.o main.o
$(CC) Graf.o wgraf.o wnazwe.o menu.o Stos.o TabDyn.o komunikaty.o main.o -o program
Graf.o: graf_struktura/Graf.cpp graf_struktura/Graf.h lifo/TabDyn.hpp lifo/Stos.hpp
$(COMP) graf_struktura/Graf.cpp -o $@
wgraf.o: wczytywanie_grafu/wczytaj_graf.cpp wczytywanie_grafu/wczytaj_graf.h graf_struktura/Graf.h lifo/Stos.hpp
$(COMP) wczytywanie_grafu/wczytaj_graf.cpp -o $@
wnazwe.o: wczytywanie_grafu/wczytaj_nazwe_pliku.cpp wczytywanie_grafu/wczytaj_nazwe_pliku.h wczytywanie_grafu/komunikaty.cpp
$(COMP) wczytywanie_grafu/wczytaj_nazwe_pliku.cpp -o $@
komunikaty.o: wczytywanie_grafu/komunikaty.cpp
$(COMP) wczytywanie_grafu/komunikaty.cpp
menu.o: menu/menu.cpp graf_struktura/Graf.h
$(COMP) menu/menu.cpp
main.o: main.cpp wczytywanie_grafu/wczytaj_nazwe_pliku.h wczytywanie_grafu/wczytaj_graf.h menu/menu.cpp graf_struktura/Graf.h
$(COMP) main.cpp
#Here is the issue.
Stos.o: lifo/Stos.hpp lifo/Stos.tpp lifo/TabDyn.hpp
$(COMP) lifo/Stos.hpp -o $@
TabDyn.o: lifo/TabDyn.hpp lifo/TabDyn.tpp
$(COMP) lifo/TabDyn.hpp -o $@
编译器实际上创建了“Stos.o”,但未能将其包含在“程序”中,从而产生引用错误。
希望现在我的问题更清楚了。抱歉,如果 makefile 不够优雅。我还是 makefile 的初学者,请理解。
最佳答案
如果你的 MyClass.h 看起来像
#ifndef MYCLASS_H
#define MYCLASS_H
template<typename T>
struct MyClass
{
void f();
};
#include "MyClass.tcc"
#endif
MyClass.tcc 文件看起来像
template<typename T>
void
MyClass::f(T)
{ }
然后使用该模板的代码(例如在名为 main.cpp 的文件中)如下所示:
#include "MyClass.h"
int main()
{
MyClass<int> m;
m.f(1);
}
Makefile 可能如下所示:
main: main.o
$(CXX) $^ $(LDFLAGS) -o $@
main.o: main.cpp MyClass.h MyClass.tcc
$(CXX) $(CXXFLAGS) $< -o $@
makefile 不应编译模板头文件,它将 main.cpp 文件编译为 .o,并将头文件列为依赖项。如果 header 或 main.cpp
发生更改,则 main.o
将重新编译。但不要编译 header 。
FWIW,在我自己的代码中,我使用 .cc
作为源文件,使用 .h
作为 header ,使用 .tcc
作为内联定义。 GCC 知道 .tcc
是一个头文件,如果您说 g++ foo.tcc
,它会正确对待它(作为头文件),请参阅 http://gcc.gnu.org/onlinedocs/gcc/Overall-Options.html GCC 识别的文件扩展名。
注意您可以使用 GCC 生成列出依赖项的 makefile 片段,而不是手动列出所有 header 依赖项,例如对于 main.cpp
您将生成一个 main.d
并在 makefile 中执行 include main.d
...这超出了本文的范围回答一下。
关于c++ - makefile 和多文件 C++ 模板问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10693712/