c++++中不能重载的运算符包括:1. 成员访问运算符(.)2. 作用域解析运算符(::)3. sizeof运算符 4. 条件运算符(?:)5. 成员指针访问运算符(.*),这些运算符具有固定的语言语义,不允许用户自定义行为,例如成员访问运算符用于直接访问对象成员,若允许重载将破坏语言机制。
在c++中,运算符重载是一个非常实用的特性,它允许我们为自定义类型(比如类或结构体)赋予已有运算符新的含义。但它的使用并不是完全自由的,有一些限制需要注意。
1. 哪些运算符不能被重载
并不是所有的运算符都可以重载。C++中有一小部分运算符是不允许重载的,主要包括:
- . 成员访问运算符
- :: 作用域解析运算符
- sizeof 获取对象大小
- ?: 条件运算符
- .* 成员指针访问运算符
这些运算符有其特定的语言语义,不允许用户自定义行为。例如,. 运算符直接用于访问对象的成员变量或函数,如果允许重载,会破坏语言的基本机制。
立即学习“C++免费学习笔记(深入)”;
2. 重载时必须遵循的一些规则
即使可以重载的运算符,也有一些基本规则需要遵守:
- 至少有一个操作数是用户定义类型的(也就是说不能重载两个内置类型之间的运算符)
- 不能改变运算符的优先级和结合性
- 不能改变运算符的操作数个数
举个例子,如果你尝试重载 + 运算符来支持两个 int 类型的加法,这是不允许的,因为它们都不是用户定义类型。
还有一个常见误区是:虽然你可以重载 && 和 ||,但这样做会导致短路行为失效,因为重载后的运算符会像普通函数一样调用,而不是保留原有的逻辑判断机制。
3. 常用运算符重载示例
下面是一些常见的运算符重载用法,方便理解实际应用。
重载 + 运算符
class Point { public: int x, y; Point(int x = 0, int y = 0) : x(x), y(y) {} // 重载 + 运算符 Point operator+(const Point& other) const { return Point(x + other.x, y + other.y); } };
这样你就可以写 Point a(1,2), b(3,4); Point c = a + b;。
重载
这个常用于调试输出:
ostream& operator<<(ostream& os, const Point& p) { os << "(" << p.x << ", " << p.y << ")"; return os; }
之后可以直接 cout
重载赋值运算符 =
如果你的类涉及资源管理(如动态内存),就需要自己实现拷贝赋值运算符:
class MyString { char* data; public: MyString& operator=(const MyString& other) { if (this == &other) return *this; delete[] data; data = new char[strlen(other.data) + 1]; strcpy(data, other.data); return *this; } };
4. 注意事项与建议
- 保持语义清晰:不要为了炫技而重载运算符,比如给 + 赋予减法的行为,这会让代码难以维护。
- 考虑是否作为成员函数还是友元函数:通常一元运算符作为成员函数更合适,二元运算符如果希望两边都能接受隐式转换,最好用非成员函数。
- 处理自赋值问题:特别是重载赋值运算符时,要检查是否是给自己赋值。
- 一致性很重要:如果你重载了 ==,也应该提供 !=;重载了
基本上就这些,C++中的运算符重载功能强大但也容易出错,尤其在资源管理和语义一致性方面,稍不注意就会埋下隐患。掌握好基本规则和典型用法,才能写出清晰、安全的代码。