下列是重载为非成员函数的运算符函数原型,其中错误的是( )。
A.Fraction operator +(Fraction, Fraction);
B.Fraction operator -(Fraction);
C.Fraction& operator =(Fraction&, Fraction);
D.Fraction& operator +=(Fraction&, Fraction);
标准答案选C,但是对于答案的解析,网上的解释众说纷纭、说法不一,这里给出我较为认同的解析。
B选项常被误选,但是“-”当负号用的话,则B选项正确。
为什么C选项错误,是因为C++强制规定,重载赋值运算符函数,只能是类的非静态的成员函数。
究其原因,当程序没有显式地提供一个以本类或本类的引用为参数的赋值运算符重载函数时,编译器会自动提供一个。现在,假设C++允许将赋值运算符重载函数定义为友元函数并且我们也确实这么做了,而且以类的引用为参数。与此同时,我们在类内却没有显式提供一个以本类或本类的引用为参数的赋值运算符重载函数。由于友元函数并不属于这个类,所以,此时编译器一看,类内并没有一个以本类或本类的引用为参数的赋值运算符重载函数,所以会自动提供一个。此时,我们再执行类似于str2=str1这样的代码,那么,编译器是该执行它提供的默认版本呢,还是执行我们定义的友元函数版本呢?
为了避免这样的二义性,C++强制规定,赋值运算符重载函数只能定义为类的成员函数,这样,编译器就能够判定是否要提供默认版本了,也不会再出现二义性。
感谢网友“同勉共进”的精彩总结。