运算符声明成类成员还是声明独立友元函数建议准则:
C++规定:
- 赋值运算符=、数组下标运算符[]、函数调用运算符()、成员访问运算符->在重载时必须声明为类的成员函数
- 流运算符<<、>>、类型转换运算符不能定义为类的成员函数,只能是 友元函数
- 一元运算符和复合赋值运算符重载时,一般声明类的 成员函数
- 二元运算符在运算符重载时,一般声明为 友元函数
注意:
- 对于很多运算特来说,可以选择使用成员函数或非成员函数来实现运算符重载,一般来说,非成员函数应该是友元函数,这样才能直接访问类的私有数据。
- 在定义运算特时,必须选择其中的一种格式,而不能同时选择这两种格式,同时定义这两种格式将被被为二义性错误,导致编译错误
- 那么哪种格式最好呢?对于某些运算特来说,成员函数是唯一合法的解释,在其他情况下,这两种格式没有太大的区别,有时,根据类的设计,使用非成员函数版本可能更好(尤其是为类定义类型转换时).
友元函数在定义时不需要使用域运算符 ::
重载流运算符后可以很容易控制某个对象的打印格式
类的友元函数是定义在类外部,但有权访问类的所有私有(private)成员和保护(protected)成员。尽管友元函数的原型有在类的定义中出现过,但是友元函数并不是成员函数。
友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类,在这种情况下,整个类及其所有成员都是友元。
如果要声明函数为一个类的友元,需要在类定义中该函数原型前使用关键字 friend,如下所示:
class Box
{
double width;
public:
double length;
friend void printWidth( Box box );
void setWidth( double wid );
};
#include <iostream>
using namespace std;
class Box
{
double width;
public:
friend void printWidth( Box box );
void setWidth( double wid );
};
// 成员函数定义
void Box::setWidth( double wid )
{
width = wid;
}
// 请注意:printWidth() 不是任何类的成员函数
void printWidth( Box box )
{
/* 因为 printWidth() 是 Box 的友元,它可以直接访问该类的任何成员 */
cout << "Width of box : " << box.width <<endl;
}
// 程序的主函数
int main( )
{
Box box;
// 使用成员函数设置宽度
box.setWidth(10.0);
// 使用友元函数输出宽度
printWidth( box );
return 0;
}
运行结果:
Width of box : 10