C++结构异常的清晰向量

C++ clear vector of structs exception

本文关键字:清晰 向量 异常 结构 C++      更新时间:2023-10-16

我定义了一个名为 Part 的结构:

struct Part
{
std::string id;
double area;
double volume;
bool operator==(const Part& part) const
{
return (area == part.area) && (volume == part.volume);
}
bool operator!=(const Part& part) const
{
return !operator==(part);
}
bool operator<(const Part &part) const
{
return (area + volume) < part.area + part.volume;
}
};

我创建了一个零件矢量并将矢量保存在二进制文件中。 然后我将二进制文件读入向量。我确认所有零件都已正确读取到矢量中。我读取二进制文件的函数是:

static bool restore_vector_from_file(const char* fileName, std::vector<Part> &parts) {
std::ifstream instream(fileName, std::ios::in | std::ios::binary);
if (!instream) {
return false;
}
instream.seekg(0, instream.end);
int bytes = instream.tellg();
instream.seekg(0, instream.beg);
int vector_size = bytes / sizeof(Part);
parts.resize(vector_size);
instream.read(reinterpret_cast<char*>(parts.data()), bytes);
if (!instream) 
{
return false;
}
instream.close();
return true;
}

我的主要功能是:

int main(int argc, char *argv[])
{
if (argc != 3)
{
std::cout << "Number of arguments must be 3" << std::endl;
return 1;
}
// Load parts with properties to a vector
const char* folderName = argv[1];
const char* vectorFileName = argv[2];
std::vector<Part> partArray;
// Load from pre-saved file to save time
if (!restore_vector_from_file(vectorFileName, partArray))
{
std::cout << "Failed to retore vector from file" << std::endl;
return 1;
}
partArray.clear();
return 0;
}

一切正常,直到我尝试清除partArray.它抛出一个异常:

Exception thrown: read access violation.
_Pnext was 0x1BC586CCB48.

即使我删除了partArray.clear(),在return 0;时也会抛出相同的异常

请帮忙。

您的结构Part不是 POD 类型,也不是微不足道的,因此您无法像现在这样保存或加载它。

以它id成员为例,std::string对象通常包含一个指针,该指针是保存的内容。该指针在任何其他进程中都无效。

您需要使用序列化来保存和加载结构。

instream.read(reinterpret_cast<char*>(parts.data()), bytes);

这是一个很大的代码气味和UB。std::string包含一个指向堆分配字符串(模 SSO)的指针,但是当您尝试reinterpret_cast它的形式文件时(假设您以相同的方式保存它),您只会得到一些垃圾值(因为它现在不指向任何有效的内存位置),因此当std::string尝试在析构函数中释放它时,它会像您经历过的那样失败。你根本无法在这里做你想做的事情。

我建议你重载结构istream& operator>> (istream& is, Part& p);。以及ostream的相应<<。这应该很容易,因为所有Part字段都已经存在这样的重载。