1

可能重复:
C++ 模板,未定义的引用

我有一个非常简单的程序,由三个文件组成,它从普通数组构建向量:

//create.hpp

#ifndef CREATE_HPP_
#define CREATE_HPP_

#include <vector>

using namespace std;

template<class T>
vector<T> create_list(T uarray[], size_t size);

#endif /* CREATE_HPP_ */

//create.cpp

#include "create.hpp"

template<class T>
vector<T> create_list(T uarray[], size_t size){
    vector<T> ulist(uarray, uarray + size);
    return ulist;
}

//main.cpp

#include <vector>
#include <iostream>

#include "create.hpp"

using namespace std;


int main(){
    char temp[] = { '/', '>' };
    vector<char> uvec = create_list<char>(temp, 2);

    vector<char>::iterator iter=uvec.begin();
    for(;iter != uvec.end();iter++){
        cout<<*iter<<endl;
    }

    return 0;
}

构建过程如下:

g++ -O0 -g3 -Wall -c -fmessage-length=0 -o create.o create.cpp
g++ -O0 -g3 -Wall -c -fmessage-length=0 -o main.o main.cpp
g++ -o main.exe main.o create.o

在构建程序时,我收到此错误:

main.o: In function `main':
../main.cpp:18: undefined reference to `std::vector<char, std::allocator<char> > create_list<char>(char*, unsigned int)'

这个程序真的很简单。但是,编译成功通过,但链接失败。然后我将所有代码移动到一个文件中,一切都像一个魅力。任何人都可以帮我解决这个问题吗?

4

2 回答 2

6

是的。答案很复杂。它与模板在 C++ 中的实际工作方式有关。

简短的回答:完整的模板定义必须在头文件中,或者您必须有一个明确的实例化(例如http://msdn.microsoft.com/en-us/library/by56e477(v=vs.80).aspx)对于 CPP 文件中的给定类型。

长答案(原因):模板不是可以编译成二进制(对象)的代码。它们只是“创建代码的秘诀”,并且只能在实例化过程中创建代码。这也是为什么不准确地使用模板可能会导致编译时间长并且二进制文件比需要的大的原因。

于 2012-12-22T17:44:34.887 回答
4

您已在单独的.cpp文件中定义模板。每个文件都是单独编译的。您create.cpp不包含任何正在运行的代码,因此它被编译器丢弃。后来,在链接阶段,当链接器尝试链接 with 的二进制文件时main.cppcreate_list它在其他编译对象中找不到它,因此您收到此错误。要解决它,您需要在您的模板中至少实例化一次,create.cpp或者在头文件中实现。

于 2012-12-22T17:48:54.120 回答