在一般友元函数的前面加上 template<typename T),注意在函数的声明和定义处都要加这个模板
例如:
//模板类,长方体类templateclass Rectangle{ ///*重载基类的输入输出运算符*/ //template //friend std::istream& operator >> (std::istream& in, Rectangle & item); //template //friend std::ostream& operator << (std::ostream& out, const Rectangle & item);protected: //受保护的成员 ElementType length; //长方体的长,在类内部可以直接使用模板定义的数据类型 ElementType width; //长方体的宽 RectangleNo myNo; //长方形的序号,嵌套类对象作为Rectangle类的一个数据成员};
友元函数的定义部分也要加上template<typename T>这个模板
/*重载输入运算符*/templatestd::istream& operator >> (std::istream& in, Rectangle & item){ in >> item.length >> item.width >> item.myNo.no; return in;}/*重载输出运算符*/template std::ostream& operator << (std::ostream& out, const Rectangle & item){ out << "length = " << item.length << " width = " << item.width << " No = " << item.myNo.no; return out;}
如果不知道怎么声明模板类的友元函数,可以在类内部用函数实现你想用友元函数实现的功能,
然后在类外的友元函数中调用这个类内函数,就可以达到相同的效果
例如:上面那个输入输出的友元函数可以改成:
//模板类,长方体类templateclass Rectangle{public: //读取长方形 void read(std::istream& in); //输出长方形 void display(std::ostream& out)const ;protected: //受保护的成员 ElementType length; //长方体的长,在类内部可以直接使用模板定义的数据类型 ElementType width; //长方体的宽 RectangleNo myNo; //长方形的序号,嵌套类对象作为Rectangle类的一个数据成员};
在类外实现read和display函数后,在类外的友元函数中调用这两个函数即可
templatevoid MyRectangle ::read(std::istream& in){ cout << "Please enter the length and width of " << myNo.no << endl; in >> length >> width;}template void MyRectangle ::display(std::ostream& out) const{ out << "the " << myNo.no << " Rectangle's length = " << length << " width = " << width;}template std::ostream& operator << (std::ostream& out, const Rectangle & item){ item.display(out); return out;}template std::istream& operator >> (std::istream& in, Rectangle & item){ item.read(in); return in;}
派生类友元函数对基类的成员使用情况:
派生类友元函数可以使用基类中非private成员,可以使用基类的protected和public成员
基类的成员函数(包括友元函数)只能操作派生类对象中基类那部分。
如果派生类中有函数和基类同名,那么派生类对象会调用派生类中的那个函数,而不是对象的基类部分调用基类函数,派生类部分调用派生类函数。