当试图重载运算符==时,如何比较两个类的私有变量

How to compare private variables of two classes when trying to overload operator==?

本文关键字:两个 变量 比较 运算符 何比较 重载      更新时间:2023-10-16

我正在尝试重新创建字符串类,但在比较这两个类时遇到了困难,就好像它们是字符串一样。这就是我在头文件中的内容:

class Mystring {
public:
// ...
private:
// this variable stores the memory address of the text
char* ptr_buffer
// ...
};

在源文件中,我有thsi代码:

bool operator== (const Mystring& a, const Mystring& b) {
return a.ptr_buffer == b.ptr_buffer;
}

但出于某种原因,当我试图比较main.cpp中的两个Mystring类时,它说它无法访问私有变量ptr_buffer。我以为它可以访问那个?如果我唯一关心比较的东西是隐藏的,我该如何比较这两者?

从C++语言的角度来看,在全局范围内引入的重载运算符与全局范围内的任何其他函数一样(除了它自定义一个内置运算符的工作方式之外)。正如普通的旧自由函数不能访问类的private字段一样,自由函数重载运算符也不能访问它为其重载运算符的类的private字段。

这样做有充分的理由。例如,自定义类重载operator <<非常常见,其中一个参数是ostream引用,另一个是自定义类型。如果这个自定义重载可以访问ostream类型的字段,那就太糟糕了!

您在类的实现文件中定义了operator ==这一事实也无关紧要。C++没有类的"实现文件"的概念,尽管类通常是这样实现的。在这方面,它将所有源文件视为平等的。

要解决此问题,可以将operator ==函数声明为类的friend

class MyString {
public:
...
friend bool operator== (const MyString& lhs, const MyString& rhs);
}
...
bool operator== (const MyString& lhs, const MyString& rhs) {
// You were given explicit authorization to access private fields,
// so go right ahead!
}

不过,独立地说,你确定你对operator ==的实现是正确的吗?您比较的是存储在MyString对象中的指针,而不是它们所指向的字符串的内容。如果您在内存中的某个位置有同一字符串文字的两个不同副本,则可能有两个逻辑相等的MyString(它们表示同一字符串),但比较起来不相等。

事实上,您可能想问一个单独的问题:是否可以仅使用MyString类型的公共接口来实现operator==在一个好的字符串类型中,答案应该是"是",因为你应该能够按顺序访问字符并查看长度。从效率的角度来看,这并不理想,但这是一个很好的测试,可以看看接口是否良好。因此,您可能希望尝试在不使用私有字段的情况下实现此函数,以确保可以,然后考虑将operator ==设置为friend的好处,以及从性能角度来看这是否是一个好主意。