我正在寻找一种优雅的方式将文本文件读入2x2向量。我使用这种方法将数据写入文本文件:
ofstream FILE(theFile, ios::out | ofstream::binary);
for(const auto& vt : data) {
std::copy(vt.begin(), vt.end(), std::ostream_iterator<double>(FILE, " "));
FILE << "\n\n";
}
我的问题是,有没有办法以类似的方式使用std::istream_iterator
来读取文本文件的内容?
编辑:
数据:
std::vector<std::vector<double> > data;
答案 0 :(得分:5)
由于您坚持使用std::istream_iterator
,这是一种方式
#include <sstream>
//....
std::vector<double> v ;
std::string str ;
while ( std::getline(FILE, str) )
{
std::stringstream ss(str);
std::copy( std::istream_iterator<double>(ss),
std::istream_iterator<double>(),
std::back_inserter(v)
) ;
data.push_back( v ); // data is your vector of vector
v.clear( );
}
答案 1 :(得分:1)
您可以创建自己的迭代器,使用向量作为基础类型。它可以与std::istream_iterator
:
#include <vector>
#include <sstream>
#include <string>
#include <algorithm>
#include <iterator>
template <
class Value,
class Container = std::vector<Value>,
class charT = char,
class traits = std::char_traits<charT>
>
class line_iterator
: public std::iterator<std::input_iterator_tag, std::vector<Value>>
{
public:
using value_type = Value;
using container_type = Container;
private:
std::basic_istream<charT, traits>* is_;
std::basic_string<charT, traits> line;
std::basic_istringstream<charT, traits> str;
container_type temp;
int count = -1;
public:
line_iterator(std::basic_istream<charT, traits>& is,
int count = -1) noexcept
: is_(&is), count(count)
{ this->read(); }
constexpr line_iterator() noexcept : is_(nullptr)
{ }
constexpr line_iterator(const line_iterator& rhs) noexcept
: is_(rhs.is_), temp(rhs.temp), count(rhs.count)
{ };
constexpr container_type operator*() const noexcept
{
return std::move(temp);
}
line_iterator& operator++()
{
this->read();
return *this;
}
line_iterator operator++(int)
{
line_iterator copy(*this);
++*this;
return std::move(copy);
}
friend constexpr bool operator==(const line_iterator& lhs,
const line_iterator& rhs) noexcept
{
return lhs.is_ == rhs.is_;
}
friend constexpr bool operator!=(const line_iterator& lhs,
const line_iterator& rhs) noexcept
{
return !(lhs == rhs);
}
private:
void read()
{
temp.clear();
if (is_)
{
if (std::getline(*is_, line))
{
str.str(line);
if (count == -1)
temp.assign(
std::istream_iterator<Value, charT, traits>{str}, {});
else if (0 <= count)
std::copy_n(
std::istream_iterator<Value, charT, traits>{str},
count,
std::back_inserter(temp));
}
if (!*is_)
is_ = nullptr;
str.clear();
}
}
};
template <
class Value,
class Container = std::vector<Value>,
class charT = char,
class traits = std::char_traits<charT>
>
line_iterator<Value, Container, charT, traits>
constexpr line_iter(std::basic_istream<charT, traits>& is,
int count = -1) noexcept
{
return line_iterator<Value, Container, charT, traits>{is, count};
}
template<class Value>
constexpr line_iterator<Value> line_iter() noexcept
{
return line_iterator<Value>{};
}
现在您可以使用以下内容初始化矢量:
std::vector<std::vector<double>> v{line_iter<double>(FILE), line_iter<double>()};