繁体   English   中英

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

[英]Writing contents of vector to file as binary data

我有一些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));

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

实际上,编译器总是抱怨将begin()的结果强制转换为char *。

确保将std::vector的元素存储在连续的内存中,因此,只要myClassTypePOD类型 ,则您可以尝试执行此myClassType 否则,您将不得不遍历std::vector将每个元素序列化为一个平面格式(根据需要)。

但是, std::vector::begin()方法将返回第一个元素的迭代器 ,而不是指向元素数据的指针 迭代器就像一个指针,但不是指针。 容器实现被允许使用实际的指针作为迭代器,但是它们使用包装器类的次数更多。 不要依靠这种行为! 以预期的方式使用迭代器API,并且不要假定迭代器是实际的指针。

要获得指向向量的第一个元素的数据的有效指针,您需要执行以下任一操作:

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

     myVect.data() 

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

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

     &myVect[0] 

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

     bool OK = myVect.empty() || File.write((char*) &myVect[0], myVect.size() * sizeof(myClassType)); 
  3. 与#2相同,但使用std::vector::front()代替:

     &myVect.front() 

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

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

     &(*(myVect.begin())) 

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

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

雷米·勒博的答案是正确的。 要添加它,如果您想增加代码的安全性,还应该使用static_assert来确保myClassType可以安全编写,如您所愿:

#include <type_traits>
static_assert(std::is_pod_v<myClassType>, "myClassType is potentialy unsafe to write as-is");

如果以后有人以不安全的方式修改myClassType使其无法在File.write()代码中使用,则此编译时检查将捕获它。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM