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

软件发布|下载排行|最新软件

当前位置:首页IT学院IT技术

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

qq_42766764   2022-11-12 我要评论

前言

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

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

Copyright 2022 版权所有 软件发布 访问手机版

声明:所有软件和文章来自软件开发商或者作者 如有异议 请与本站联系 联系我们