前言
总所周知,C++的std::vector<bool>
并不是一种“标准”的容器。该容器按位存储数据,使用at(size_t)
或者其重载的operator[](size_t)
返回的都是一个特化的Reference类,使用begin()
之类的函数也是特殊的迭代器。而且不同的编译器,其标准库的实现方式也不一样。如此,直接将数据std::vector<bool>
转储到文件似乎就显得不可能了。那么是否有方法可以进行转储呢?答案是有的,只要能找到存储数据的起始指针即可将数据转储。
获取数据源地址
MSVC
1、微软没有实现data()
函数的接口。
2、微软直接暴露(public)了存储std::vector<bool>
的std::vector<unsigned int>
。
3、微软的迭代器直接暴露(public)了迭代器指向的数据指针。
GCC
1、GCC偏特化实现了data()
函数接口,但返回是void。
2、GCC提供了访问直接存储数据的一个结构化表述类的接口,但真的很不优雅。
3、GCC的迭代器同样直接暴露了迭代器指向的数据指针。
数据地址获取方法
auto GetBoolVectorStartAddress(std::vector<bool>& vec) {
#ifdef __GNUC__
/*方法一
auto begin = vec.begin();
return begin._M_p;
*/
//方法二
auto Impl = vec._M_get_Bit_allocator(); //获取_Bvector_impl类型的_M_impl;
return Impl._M_start._M_p; //Impl._M_start就是begin返回的迭代器
#else
/*方法一
auto& source = vec._Myvec;
return &source[0];*/
//方法二
auto begin = vec.begin();
return begin._Myptr;
#endif
}
#include<fstream>
int mian(){
std::vector<bool> test;
for(int i = 0; i < 65536; i++)
{
test.push_back(i % 2 ? true : false);
}
auto StartAddress = GetBoolVectorStartAddress(test);
std::ofstream ofs("test.bin", std::ios::binary|std::ios::out);
ofs.write((char*)StartAddress, 8192);
ofs.close();
return 0;
}
结果
总结
将std::vector<bool>
转储文件的方法很简单,只要找到相应的起始位置的指针,在将数据直接使用流输出即可。