[英]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
的元素存储在连续的内存中,因此,只要myClassType
是POD类型 ,则您可以尝试执行此myClassType
。 否则,您将不得不遍历std::vector
将每个元素序列化为一个平面格式(根据需要)。
但是, std::vector::begin()
方法将返回第一个元素的迭代器 ,而不是指向元素数据的指针 。 迭代器就像一个指针,但不是指针。 容器实现被允许使用实际的指针作为迭代器,但是它们使用包装器类的次数更多。 不要依靠这种行为! 以预期的方式使用迭代器API,并且不要假定迭代器是实际的指针。
要获得指向向量的第一个元素的数据的有效指针,您需要执行以下任一操作:
使用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));
使用std::vector::operator[]
获取对第一个元素的引用,然后使用operator&
获取元素的地址:
&myVect[0]
请注意, operator[]
不执行边界检查,因此如果指定的索引为> = size()
,则将返回未定义的指针,因此请确保在写入向量之前检查向量是否为空:
bool OK = myVect.empty() || File.write((char*) &myVect[0], myVect.size() * sizeof(myClassType));
与#2相同,但使用std::vector::front()
代替:
&myVect.front()
请注意,在空向量上调用front()
是未定义的行为,因此请确保检查向量是否为空:
bool OK = myVect.empty() || File.write((char*) &myVect.front(), myVect.size() * sizeof(myClassType));
使用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.