3

我有一个类,在头部定义为:

template <typename T> class MyClass
{
   template <typename U> friend std::ostream& operator<<(std::ostream& output, const MyClass<U>& p);
   public:
      ...
}

在一个实现文件中,我有:

template <typename U> std::ostream& operator<<(std::ostream& output, const MyClass<U>& m)
{
   output << "Some stuff";
   return output;
}

这一切看起来都很犹太。但是,当我尝试使用此运算符(即 std::cout << MyClass())时,我收到以下链接器错误:

Undefined symbols: std::basic_ostream<char, std::char_traits<char> >& operator<< <InnerType>(std::basic_ostream<char, std::char_traits<char> >&, MyClass<InnerType> const&)

我很惊讶编译器没有自动为我生成这个......关于我做错了什么的任何建议?

4

2 回答 2

8

在一个实现文件中,我有:

那就是问题所在。您不能在头文件和实现文件之间拆分模板定义。由于模板的性质,C++ 编译器在这里很挑剔。定义标题中的所有代码以使其工作。

事实上,这里的问题是所有模板定义必须驻留在同一个编译单元中,因为 C++ 标准没有定义模板信息如何在不同单元之间共享。这些单元由链接器拼接在一起,但泛型是在编译时(更早)而不是在链接时解析的。

从理论上讲,C++ 标准定义了一个关键字export, 来处理这些情况。实际上,没有编译器实现这一点(有一个例外?),并且无意改变这一点,因为成本/实用性的权衡被认为不够好。

于 2009-02-15T16:31:02.290 回答
1

模板太多 - 这有效:

#include <iostream>
using namespace std;
template <typename T> struct MyClass {

    friend ostream & operator << ( ostream & os, MyClass<T> & c ) {
        os << "MyClass\n";
        return os;
    }
};

int main() {
    MyClass <int> c;
    cout << c;
}
于 2009-02-15T16:37:12.807 回答