C 语言

关注公众号 jb51net

关闭
首页 > 软件编程 > C 语言 > C++ std::vector<bool>转储文件

C++的std::vector<bool>转储文件问题

作者:qq_42766764

这篇文章主要介绍了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>转储文件的方法很简单,只要找到相应的起始位置的指针,在将数据直接使用流输出即可。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

您可能感兴趣的文章:
阅读全文