4

关于我之前的一个问题的后续问题,已经得到了完美的回答。快速回顾一下,我在创建一个包含巨大数组的类时遇到了麻烦(堆栈溢出错误)。在答案中,一些用户建议我改用 std::vector 。

读取数据的函数如下所示:

Test()
{
   memset(myarray, 0, sizeof(myarray));
   FILE* fstr = fopen("myfile.dat", "rb");
   size_t success= fread(myarray, sizeof(myarray), 1, fstr);
   fclose(fstr);
}

对于一个看起来像这样的 myarray:

int myarray[45000000];

我的问题是:我怎样才能把它读成一个更可取的:

std::vector<int> myvector;

我搜索了谷歌,找到了多个答案,通常指向以下代码:

std::ifstream input("myfile.dat", std::ios::in | std::ifstream::binary);
std::copy(std::istream_iterator<int>(input), 
     std::istream_iterator<int>(), 
     std::back_inserter(myvector));

在实现这个之后,当调用 myvector.size() 时,我得到 16(无论出于何种原因),并且访问向量元素会导致超出向量边界的立即崩溃。

那么我该怎么做才能做到这一点呢?我曾经在某处读到我可以简单地使用“旧”方法,然后将数组读入向量,但这似乎违背了首先使用向量的目的。

4

2 回答 2

6

fread()读取您的文件binary,同时ifstream_iterator尝试提取格式化的整数(如42)。

你想要resize你的vector并使用input.read(...)

const size_t size = 45000000; // change this to the appropriate value
std::vector<char> myvector(size, 0);
std::ifstream input("myfile.dat", std::ios::in | std::ifstream::binary);
input.read(&myvector[0], myvector.size());

请注意,您需要使用 astd::vector<char>因为read期望第一个参数是 a char *T如果正确转换类型,则可以使用其他类型:

input.read(reinterpret_cast<char*>(&myvector[0]), myvector.size() * sizeof(T));
于 2013-02-28T19:11:12.347 回答
4

如果您使用 C++,您应该尽量避免同时使用 C FILEAPI——这样您就走在了正确的轨道上。您遇到的问题是istream_iterator将输入读取为文本,而不是二进制 - 它需要 ASCII 数字。这不是:

std::vector<int> vec(45000000);

std::filebuf fb;
fb.open("myfile.dat", std::ios_base::in | std::ios_base::binary);
fb.sgetn((char*)&vec[0], vec.size() * sizeof(vec[0]));
于 2013-02-28T19:17:06.580 回答