23 宁以 non-member、non-friend 替换member函数,可增加封装性、包裹弹性和机能扩充性
- non-member non-friend 函数相对于member和friend函数而言,具有更大的封装性,因为它不能够访问class内之private成分。
- non-member 可以是另一个class 的member,如工具类。
- C++中比较自然的做法是让non-member成为与class同一命名空间中的non-member函数,常称为便利函数。
- 当某个class拥有大量的便利函数时,可将便利函数分类,然后根据类别将便利函数声明到不同的头文件中,这样使用者只需要包含感兴趣的功能。C++标准库就是这样的组织方式,如仅想使用vector功能只需
#include <vector>
- 将所有便利函数放在多个头文件内但隶属于同一个命名空间,意味着客户可以轻松扩展这一组便利函数。
24 若所有参数皆需类型转换,请为此采用non-member函数
一、member 函数的问题
class Rational{
public:
Rational(int numerator = 0,int denominator = 1);
int numerator() const;
int denominator() const;
const Rational operator*(const Rational& rhs) const;
private :
...
};
Rational oneEight(1,8);
Rational oneHalf(1,2);
Rational result = oneEight * oneHalf;
result = oneHalf * 2; // 发生了隐式转换
result = 2 * oneHalf; // 错误
只用当参数被列于参数列内,这个参数才是隐式类型转换的合格参与者。
二、改用非成员函数
const Rational oprator*(const Rational& lhs,const Rational rhs){
return Rational(lhs.numerator() * rhs.numerator(),lhs.denominator()*rhs.denominator());
}
Rational oneForth(1,4);
Rational result = oneForth * 2; // 发生了隐式转换
result = 2 * oneForth; // 正确
- member函数的反面是non-member 函数,不是friend 函数。
- 无论何时如果你可以避免friend 函数就该避免。
- 当跨进Template C++ 并让Rational 成为一个class template 而非class,又有需要考虑的新争议、新解法。