重载操作程序时出错>>用于类中的字符串 memebr

Error when overloading opertor>> for string memebrs in the class

本文关键字:gt 字符串 memebr 用于 出错 程序 重载 操作      更新时间:2023-10-16

当我输入字符串类型的值时,例如:

_ho我键入了Peter
_hoten我键入了Peter Parker
_ten我键入Marry

我在屏幕上的输出是:

PeterPeterMarry

这是我的代码:

class SinhVien
{
private:
string _ho;
string _tenlot;
string _ten;
public:
static int InstanceCount;
SinhVien();
string ToString() const;
friend istream& operator>>(istream& in, SinhVien* p);
friend ostream& operator<<(ostream& out, const SinhVien* p);
~SinhVien();
};
istream& operator>>(istream& in, SinhVien *p)
{
cout << "Nhap ho: n";
in >> p->_ho;
rewind(stdin);
cout << "Nhap ten lot: n";
in >>  p->_tenlot;
rewind(stdin);
cout << "Nhap ten: n";
in >> p->_ten;
return in;
}
string SinhVien::ToString() const
{
stringstream writer;
writer << _ho << " " << _tenlot << " " << _ten << "n";
return writer.str();
}
ostream& operator<<(ostream &out, const SinhVien* p)
{
out << p->ToString();
return out;
}
void main()
{
SinhVien *a;
a = new SinhVien();
cin >> a;
cout << a;
cout << "nTo string:n";
cout << a->ToString();
delete a;
system("pause");
}

std::basic_istream::operator>>重载中,您需要使用std::geline((而不是std::cin >>,这样您就可以使用空格获得完整的输入名称

其次,您应该传递对象指针的引用,这样更改将应用于传递的对象,而不是其副本。

修复:

std::istream& operator>>(std::istream& in, SinhVien* &p)
{                                                // ^^ take ref of object you pass, so that the change will be applied to the object, not to the copy of it.
std::cout << "Nhap ho: n";
std::getline(in, p->_ho);   // change
std::rewind(stdin);
std::cout << "Nhap ten lot: n";
std::getline(in, p->_tenlot);   // change
std::rewind(stdin);
std::cout << "Nhap ten: n";
std::getline(in, p->_ten);   // change
return in;
}

以上内容适用于单个输入。然而,多个输入和在main()中使用std::cin >>的情况可能再次导致一些输入跳过问题。感谢@Yksisarvien指出这一点。您可以在这篇SO文章中阅读更多内容:为什么std::getline((在格式化提取后跳过输入?


旁注:在现代C++中,您不再需要管理原始指针。因为您有智能指针,当对象超出范围时,它将管理对象的生存期。所以尽可能使用它。

这意味着你可以做这样的事情:SEE LIVE

#include <memory>
class SinhVien
{
private: // memebrs
public:
// other member functions
friend istream& operator>>(istream& in, const std::unique_ptr<SinhVien> &p);
friend ostream& operator<<(ostream& out, const std::unique_ptr<SinhVien> &p);
};
std::istream& operator>>(std::istream& in, const std::unique_ptr<SinhVien> &p)
{                                                                          
std::cout << "Nhap ho: n";
std::getline(in, p->_ho);   // change
std::cout << "Nhap ten lot: n";
std::getline(in, p->_tenlot);   // change
std::rewind(stdin);
std::cout << "Nhap ten: n";
std::getline(in, p->_ten);   // change
return in;
}
std::ostream& operator<<(std::ostream &out, const std::unique_ptr<SinhVien> &p)
{
return out << p->ToString();
}
int main()
{
auto a = std::make_unique<SinhVien>();
std::cin >> a;
std::cout << a;
std::cout << "nTo string:n";
std::cout << a->ToString();
return 0;
}