使用迭代器正确读取和写入std :: vector到文件中

wal*_*lly 3 c++ c++11

我试图理解这里提供的答案,但我似乎无法使其发挥作用.

这是我尝试过的:

#include <algorithm>
#include <iostream>
#include <iterator>
#include <vector>
#include <fstream>

int main()
{
    std::string path("numbersfile");

    std::vector<int> myVector{1,16,32,64};
    std::vector<int> newVector{};

    std::ofstream FILE(path,std::ios::out | std::ofstream::binary);
    std::copy(myVector.begin(),myVector.end(),std::ostreambuf_iterator<char>(FILE));

    std::ifstream INFILE(path,std::ios::in | std::ifstream::binary);
    std::istreambuf_iterator<char> iter(INFILE);
    //std::copy(iter.begin(),iter.end(),std::back_inserter(newVector)); //this doesn't compile
    std::copy(iter,std::istreambuf_iterator<char>{},std::back_inserter(newVector)); // this leaves newVector empty
}
Run Code Online (Sandbox Code Playgroud)

newVector在最后一次之后仍然是空的copy.怎么可以更新最后一个语句来填充newVector

wal*_*lly 10

copy调用第二个文件时,文件尚未准备好读取.(感谢Piotr Skotnicki在评论中的回答)

一个flush允许程序工作的调用:

#include <algorithm>
#include <iostream>
#include <iterator>
#include <vector>
#include <fstream>

int main()
{
    std::string path("numbersfile");

    std::vector<int> myVector{1,16,32,64};
    std::vector<int> newVector{};

    std::ofstream FILE(path,std::ios::out | std::ofstream::binary);
    std::copy(myVector.begin(),myVector.end(),std::ostreambuf_iterator<char>(FILE));
    FILE.flush(); // required here

    std::ifstream INFILE(path,std::ios::in | std::ifstream::binary);
    std::istreambuf_iterator<char> iter(INFILE);
    //std::copy(iter.begin(),iter.end(),std::back_inserter(newVector)); //this doesn't compile
    std::copy(iter,std::istreambuf_iterator<char>{},std::back_inserter(newVector)); // this leaves newVector empty
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

ofstream该时仍然在范围ifstream被创建.如果ofstream已经调用了析构函数,那么该文件也已经准备好了ifstream.在以下程序中,ifstream自动销毁:

#include <algorithm>
#include <fstream>
#include <iterator>
#include <vector>

std::string filename("numbersfile");

std::vector<double> myVector{1.342, 16.33, 32.1, 12364};

void write_vector_to_file(const std::vector<double>& myVector, std::string filename);
std::vector<double> read_vector_from_file(std::string filename);

int main()
{
    write_vector_to_file(myVector, filename);
    auto newVector{read_vector_from_file(filename)};
    return 0;
}

void write_vector_to_file(const std::vector<double>& myVector, std::string filename)
{
    std::ofstream ofs(filename, std::ios::out | std::ofstream::binary);
    std::ostream_iterator<double> osi{ofs," "};
    std::copy(myVector.begin(), myVector.end(), osi);
}

std::vector<double> read_vector_from_file(std::string filename)
{
    std::vector<double> newVector{};
    std::ifstream ifs(filename, std::ios::in | std::ifstream::binary);
    std::istream_iterator<double> iter{ifs};
    std::istream_iterator<double> end{};
    std::copy(iter, end, std::back_inserter(newVector));
    return newVector;
}
Run Code Online (Sandbox Code Playgroud)


Now*_*Man 5

您的代码中存在许多缺陷:

  1. 你定义了一个名为FILE THIS IS BAD BAD BAD的变量.FILE是已存在对象的名称,它与命名vectoras:的实例相当std::vector<int>array{}.
    它不仅令人困惑,而且非常危险,因为它几乎可以肯定地导致命名冲突.此外,所有国会大厦名称都应保留给宏.

  2. 你永远不会检查文件是否实际打开,如果不是,编译器不会警告你,并且流不会给出任何失败的指示(除非明确检查).所以,你应该经常检查.最简单的方法是使用流布尔运算符:
    if (!ifile) throw std::runtime_error("error opening file");

  3. 你写道,这不编译:

    std::copy(iter.begin(),iter.end(),std::back_inserter(newVector));

    为什么会这样?迭代器本身没有beginend函数,与迭代器关联的对象具有这些方法.

  4. 在这里拼凑所有这些是您的代码的修改版本:

    {
        std::string path("numbersfile");
    
        std::vector<int> myVector{ 1,16,32,64 };
        std::vector<int> newVector{};
    
    
        std::ofstream outfile(path, std::ios_base::binary);
        std::copy(myVector.begin(), myVector.end(), std::ostreambuf_iterator<char>(outfile));
    
        outfile.close(); 
    
        std::ifstream infile(path,std::ios_base::binary);
        std::istreambuf_iterator<char> iter(infile);
        std::copy(iter, std::istreambuf_iterator<char>(),    std::back_inserter(newVector)); // this leaves newVector empty
    
        infile.close(); // close explicilty for consistency 
    }
    
    Run Code Online (Sandbox Code Playgroud)