运算符重载是接口设计而非语法糖,应使类行为如内置类型般自然;只重载有直观语义的运算符,优先==、!=、、+、-、*、[]、()等,谨慎=、&&、||、,,禁用?:、::、.等;成员/非成员选择依转换需求与对称性而定,坚持无副作用、行为一致、显式控制隐式转换。
运算符重载不是语法糖,而是接口设计——它该让类的行为“像内置类型一样自然”,而不是炫技或绕过类型系统。用错地方反而破坏可读性、引发隐式转换陷阱,甚至导致二义性编译错误。
比如 operator+ 应表示“可交换的、无副作用的组合”,operator== 应满足自反、对称、传递;而 operator* 用于矩阵类很自然,但给一个日志类重载 operator* 就令人困惑。
核心原则:左侧操作数需隐式转换时,必须用非成员函数(常为友元);涉及对称性或需要访问私有成员时,再考虑友元。
std::string s; if (s == "hello"))return T(lhs) += rhs;),避免重复逻辑
or[] / operator()-> / operator++:必须是成员(只有成员能定义这些)用户看到 a + b,默认期待它不修改 a 或 b,返回新对象;看到 a += b,才预期 a 被修改。打破这个直觉就是bug温床。
a += b += c;)带单参数构造函数(或 conversion operator)容易引发静默转换,让重载运算符被误触发。例如:
class String { public: String(const char*); }; String s = "abc"; if (s == "def") // OK —— 但若还有 String(int),"s == 42" 就可能意外调用!解决方案:
explicit operator bool() const;
基本上就这些。运算符重载不是越多越好,而是越少、越准、越符合直觉越好。宁可多写几个命名函数(如 multiply_by()),也不要让 operator* 变成“执行某种业务逻辑”的黑盒。