转自 http://blog.sina.com.cn/s/blog_6a7439b30100vfib.html
文章
在模板中输入输出流的重载,若使用友元在类内声明,在类外实现,那么链接时就会报错,可以使用三种方式来实现输入输出流的重载。
那么输入输出流重载为什么不能在类内声明,类外实现呢??因为模板比较特殊,若果在模板类外实现重载的话:
template<class T>
ostream& operator<<(ostream& out,Test<T>& t)
{
return out<<"data is "<<t.data;
} //--------------------------------------------
上面正好是函数模板的定义,而我们知道操作符重载函数不是类的成员函数,因此此处相当于定义了一个新 的函数模板(不同于类中的friend ostream& operator<<(ostream& out,Test<T>& t) )。但若去掉template<class T> ,函数中的参数Test<T>就不知是什么类型,所以不能在模板类内声明,类外实现操作符重载。
template<class T>
ostream& operator<<(ostream& out,Test<T>& t)
{
return out<<"data is "<<t.GetData();
} //--------------------------------------------
template<class T>
istream& operator>>(istream& in,Test<T>& t)
{
T item;
in>>item;
t.SetData(item);
return in;
}//---------------------------------------------
int main()
{
Test<int> b(3);
cout<<b<<'\n';
cin>>b;
cout<<b<<'\n';
return 0;
}
三、使用过渡函数
#include "stdafx.h"
#include <iostream>
using namespace std;
template<class T>
class Test
{
public:
Test(const T& t):data(t){}
//---------------------------------------------
template<class CharT,class CharTraits>
basic_ostream<CharT,CharTraits>& Output(basic_ostream<CharT,CharTraits>& out)const //输出流过渡函数
{
return out<<"data is "<<data;
} //--------------------------------------------
template<class CharT,class CharTraits>
basic_istream<CharT,CharTraits>& Input(basic_istream<CharT,CharTraits>& in) //输入流过渡函数
{
return in>>data;
}//---------------------------------------------
private:
T data;
};//-----------------------------------------------------------------
template<class T,class CharT,class CharTraits>
basic_ostream<CharT,CharTraits>& operator<<(basic_ostream<CharT,CharTraits>& out,const Test<T>& t) //输出流重载
{
return t.Output(out);
}//------------------------------------------------------------------
template<class T,class CharT,class CharTraits>
basic_istream<CharT,CharTraits>& operator>>(basic_istream<CharT,CharTraits>& in,Test<T>& t) //输入流重载
{
return t.Input(in);
}//------------------------------------------------------------------
int main()
{
Test<int> b(4);
cout<<b<<'\n';
cin>>b;
cout<<b<<'\n';
return 0;
}