2

我有一个 C++ 模板类使用 #include 指令拆分为两个文件,如此链接中所建议的 -->模板分布在多个文件中

对于这两个文件,我使用了标准命名 *.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的初学者,请理解。

4

2 回答 2

4

如果您的 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,并将头文件列为依赖项。如果任何一个标题或main.cpp更改,main.o则将重新编译。但是不要编译标题。

FWIW,在我自己的代码中,我.cc用于源文件、.h标题和.tcc内联定义。GCC 知道这.tcc是一个头文件,如果你说它会正确对待它(作为头文件)g++ foo.tcc,请参阅http://gcc.gnu.org/onlinedocs/gcc/Overall-Options.html了解 GCC 识别的文件扩展名。

注意而不是手动列出所有头文件依赖项,您可以使用 GCC 生成列出依赖项的 makefile 片段,例如,因为main.cpp您将生成一个main.d并且在 makefile 中执行include main.d...不过,这超出了此答案的范围。

于 2012-05-21T23:55:00.103 回答
1

在您的 makefile 中既不包含 .h 也不包含 .cpp。使用该模板的文件将包括 .h(其中将包括 .cpp),并且每个翻译单元都将以模板所有功能的(弱)符号结束,而没有任何文件在您的 Makefile 中。

于 2012-05-21T23:00:39.673 回答