将向量的内容作为二进制数据写入文件

Tod*_*son 0 c++ vector

我有一些 15 年前的 C++ 代码,我正试图将它们带到更现代的时代。在这个阶段,我试图让用 Visual C++ 6.0 编译的代码现在用 VS 2003 (Microsoft Visual C++ .NET 69462-335-0000007-18915) 编译。

std::vector<myClassType> myVect;
...

// Assuming vector.begin() simply points to start of a contiguous array
bool OK = File.write((char*) myVect.begin(),
                     myVect.size() * sizeof(myClassType));
Run Code Online (Sandbox Code Playgroud)

当我阅读评论时,我很担心......我们真的可以假设向量在连续内存中吗?如果没有,有什么更好的方法可以做到这一点?我是否需要遍历向量内容并从 myClassType 写入每组数据?

事实上,编译器抱怨无论如何将 begin() 的结果转换为 char* 。

Rem*_*eau 5

std::vector保证将其元素存储在连续内存中,因此您尝试执行的操作是可以的,前提是它myClassTypePOD 类型。否则,您将不得不根据需要std::vector将每个元素一次一个地序列化为平面格式。

但是,该std::vector::begin()方法返回指向第一个元素的迭代器,而不是指向元素数据的指针。迭代器类似于指针,但不是指针。容器实现允许使用实际指针作为迭代器,但更多时候它们使用包装类代替。不要依赖这种行为!以预期的方式使用迭代器 API,不要假设迭代器是一个实际的指针。

要获得指向向量第一个元素数据的有效指针,您需要:

  1. 使用std::vector::data()(仅限 C++11 及更高版本):

    myVect.data()
    
    Run Code Online (Sandbox Code Playgroud)

    即使向量为空,这也可以安全使用。当为 0时data()可能会也可能不会返回,但这没关系,因为当为 0时不会尝试访问任何内存。nullptrsize()write()size()

    bool OK = File.write((char*) myVect.data(), myVect.size() * sizeof(myClassType));
    
    Run Code Online (Sandbox Code Playgroud)
  2. 用于std::vector::operator[]获取对第一个元素的引用,然后用于operator&获取元素的地址:

    &myVect[0]
    
    Run Code Online (Sandbox Code Playgroud)

    请注意,operator[]它不执行边界检查,因此如果指定的索引是 >= size(),则将返回一个未定义的指针,因此请确保在写入之前检查向量是否为空:

    bool OK = myVect.empty() || File.write((char*) &myVect[0], myVect.size() * sizeof(myClassType));
    
    Run Code Online (Sandbox Code Playgroud)
  3. 与 #2 相同,但std::vector::front()改为使用:

    &myVect.front()
    
    Run Code Online (Sandbox Code Playgroud)

    请注意,调用front()空向量是未定义的行为,因此请确保检查向量是否为空:

    bool OK = myVect.empty() || File.write((char*) &myVect.front(), myVect.size() * sizeof(myClassType));
    
    Run Code Online (Sandbox Code Playgroud)
  4. 使用begin()获取第一个元素的迭代器,然后使用operator*解引用迭代器来获取对它指向的元素的引用,然后就可以获取元素的地址:

    &(*(myVect.begin()))
    
    Run Code Online (Sandbox Code Playgroud)

    请注意,当向量为空时begin()返回end()迭代器,取消引用end()迭代器是未定义的行为,因此请确保检查向量是否为空:

    bool OK = myVect.empty() || File.write((char*) &(*(myVect.begin())), myVect.size() * sizeof(myClassType));
    
    Run Code Online (Sandbox Code Playgroud)