C++模板声明与实现分开(编译报错)
缘起,在我在用C++模板类写数据结构的时候会发生报错,报错如下: 部分声明如下:
void addToDLLTail(const T&);
部分实现如下:
template<class T>
void DoublyLinkList<T>::addToDLLTail(const T& el){
if (tail != 0){
tail = new DLLNode<T>(el ,0, tail);
tail->prev->next = tail;
}
else head = tail = new DLLNode<T>(el);
}
第一次写的时候,逻辑完全正确,刚开始以为我是我不熟悉C++的类模板的写法,因此特意翻了学校教材《C++数据结构与算法》,经过一番的修改后,编译仍然报错。 我也曾经以为这是编译器出现了问题,因为这是我第一次用Vs code + gcc + cmake 来编写程序,之前都用IDE,找了很久没有找到报错原因。 最后我无意中找到了报错原因,教材中的声明和实现都是在同一个.h 文件里编写的,这和我之前习惯用C语言写数据结构有很大的区别。我之前写程序,习惯于将一类声明全部写在一个.h 文件之中,将所有的实现同时写在一个.c 之中。而,在这里,声明和实现竟然同时写在.h 文件之中,效仿书上的写法,程序运行成功了。 经过我在网上搜寻了资料,原来这涉及到了cpp的编译问题。 例如,我们在刚开始用gcc来编译时,通常在终端这些写
gcc hello.c -o hello.o
这样的我话,我们才编译了一个文件,那么,如果有多个.c 文件怎么整,就牵扯到了上文所说的cmake,gcc分开编译了多个.c 文件成.o 文件后,cmake就将多个.o 链接起来,就成为了我们需要的程序。 在编译main.cpp 时,会编译.h 头文件,但是.h 头文件中只有函数声明,不进行编译,链接器在寻找函数时,出现寻找不到函数而报错,就是上面的undefined reference to 的报错。然而在编译.cpp 文件时,函数模板只进行一次编译,没有具体使用,所以没有二次编译。所以在链接main.cpp 时会发生报错。
在以后我们编写类模板时,解决这错误的办法就是将声明和实现都写在同一文件里。文件后缀可以改成.cpp 。
|