c++ - makefile 和多文件 C++ 模板问题

标签 c++ templates makefile

按照此链接中的建议,我已使用 #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/

相关文章:

c++ - 将 std::regex 用于简单的 RX 是好的做法吗?

c - 子 Makefile.am 中的标志

mysql - Make 似乎认为先决条件是中间文件,将其删除

c++ - 信号处理程序与事件处理程序

c++ - c++ 中的 malloc 和构造函数

c++ - 使用 SFINAE 检测 constexpr

c++ - 帮助双向链表?

c - 具有自动目标创建功能的潜在大型项目的 Makefile

C++ 基本字符串异常

c++ - 具有模板模板参数的CRTP派生类