C++的std::vector<bool>轉儲文件問題
前言
總所周知,C++的std::vector<bool>并不是一種“標準”的容器。
該容器按位存儲數(shù)據(jù),使用at(size_t)或者其重載的operator[](size_t)返回的都是一個特化的Reference類,使用begin()之類的函數(shù)也是特殊的迭代器。
而且不同的編譯器,其標準庫的實現(xiàn)方式也不一樣。如此,直接將數(shù)據(jù)std::vector<bool>轉儲到文件似乎就顯得不可能了。
那么是否有方法可以進行轉儲呢?答案是有的,只要能找到存儲數(shù)據(jù)的起始指針即可將數(shù)據(jù)轉儲。
獲取數(shù)據(jù)源地址
MSVC
1、微軟沒有實現(xiàn)data()函數(shù)的接口。
2、微軟直接暴露(public)了存儲std::vector<bool>的std::vector<unsigned int>。
3、微軟的迭代器直接暴露(public)了迭代器指向的數(shù)據(jù)指針。
GCC
1、GCC偏特化實現(xiàn)了data()函數(shù)接口,但返回是void。
2、GCC提供了訪問直接存儲數(shù)據(jù)的一個結構化表述類的接口,但真的很不優(yōu)雅。
3、GCC的迭代器同樣直接暴露了迭代器指向的數(shù)據(jù)指針。
數(shù)據(jù)地址獲取方法
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>轉儲文件的方法很簡單,只要找到相應的起始位置的指針,在將數(shù)據(jù)直接使用流輸出即可。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
使用VS2010創(chuàng)建MFC ActiveX工程項目
VS2010開發(fā)ActiveX有兩種方法,分別是MFC和ATL。MFC開過起來比較簡單,但是最終生成的文件比較大,ATL是專門用來開發(fā)ActiveX的,但是相對比較難,必須知道很多原理機制和API。咱先從MFC開發(fā)ActiveX開始吧。2015-06-06虛函數(shù)與純虛函數(shù)(C++與Java虛函數(shù)的區(qū)別)的深入分析
本篇文章是對虛函數(shù)與純虛函數(shù)進行了詳細的分析介紹,需要的朋友參考下2013-06-06c++中#include <>與#include""的區(qū)別詳細解析
<>先去系統(tǒng)目錄中找頭文件,如果沒有在到當前目錄下找。所以像標準的頭文件 stdio.h、stdlib.h等用這個方法2013-10-10C++入門教程詳解之命名空間、函數(shù)重載、缺省參數(shù)
這篇文章主要介紹了C++入門教程詳解之命名空間、函數(shù)重載、缺省參數(shù),本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-06-06