首页 > 解决方案 > BAD ACCESS 读入对象后

问题描述

我正在为一个学校项目制作密码管理器,该项目带有巨大的保修无效,如果在所有贴纸上都使用的话。目的是能够对程序中的数据进行 CRUD,并以“加密”形式将其写入磁盘。

我有一个基本类型和子类型的动态数组,我可以使用 Library::WriteToFiles() 将其写入内存并使用 Library::ReadFromFiles() 从内存中读取。这一切在我的调试设置中都很有效,我总是写然后立即读回内存。

int main() {
bool debug = true;
if (debug) {

    Library library = Library();
    library.Add(new Password("Outside", "yobro"));
    library.Add(new Password("Wut", "idk"));
    library.Add(new UsernameLogin("Yo", "M8", "James May"));

    library.Print();

    cout << endl;

    library.WriteToFiles();
    library.ReadFromFiles();

    library.Print();
    return 0;

}

但是当我注释掉写入调用时,我从读取函数中的这个位获得了错误的访问权限

void Library::ReadFromFiles() {
this->Clear();

ifstream baseFileObject;
ifstream usrNameFileObject;

baseFileObject.open("base.txt");
while (true) {
    if (baseFileObject.eof()) break;
    Password* password = new Password();
    baseFileObject.read((char*) password, sizeof(*password));
    if ( ! password->IsEmpty()) { // <- Trying to call IsEmpty() throws a BAD ACCESS
        Add(password);
    }
}
baseFileObject.close();

当我尝试调用 Password 类中非虚拟的其他函数时,它们运行得很好。IsEmpty() 在读取之前也可以正常运行。我检查了指针,它始终保持不变。文件正确打开,数据被读入对象。

最让我困惑的是为什么当我已经写了那个执行时它会起作用。如果读入这样的对象会破坏 vtable,那么为什么它有时会起作用?

编辑:密码类有 3 个变量。两个字符串和一个布尔值。没有指针。以下是公共功能。我只是对虚拟功能有问题。

public:
virtual int Type();
virtual string ToString();
virtual bool IsEmpty();
virtual bool Encrypt(unsigned int key);
virtual bool Decrypt(unsigned int key);

[[nodiscard]] bool isEncrypted() const;

[[nodiscard]] const string &getPassword() const;
void setPassword(const string &newPassword);

[[nodiscard]] const string &getLocation() const;
void setLocation(const string &newLocation);

int PasswordStrength();

标签: c++virtual-functionsvtable

解决方案


这条线已经很明确了:

baseFileObject.read((char*) password, sizeof(*password));

只是不适用于我的用例。序列化在这里是显而易见的、可扩展的、理想的解决方案,但由于这是一个学校项目,我将使用老式的将文本写入文件并使用 sstream 或类似的东西将其解析回来。

谢谢您的帮助 :)


推荐阅读