C++的std::vector<bool>转储文件问题
前言
总所周知,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>转储文件的方法很简单,只要找到相应的起始位置的指针,在将数据直接使用流输出即可。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
最新评论