Differences in reading from an istream vs. stringstream

D

david.crow

Given the following input file:

bob 1 2 3 4 5
mary 2 3 4 5 6 7
susan 3 4 5 6 7 8 9

This code snippet does not read it correctly:

class Student
{
public:
...
private:
std::string _Name;
std::vector<int> _Grades;
};
....
/***** this only gets called once *****/
std::istream& operator>>(std::istream& is, Student& s)
{
string name = "";
is >> name;
std::vector<int> grades;
std::copy(std::istream_iterator<int>(is),
std::istream_iterator<int>(), std::back_inserter(grades));
/***** at this point, name and the grades vector are correct for
the 'first' line in the input file *****/
s.setName(name);
s.setGrades(grades);
return is;
}
....
std::ifstream fin;
std::vector<Student> students;
std::copy(std::istream_iterator<Student>(fin),
std::istream_iterator<Student>(), std::back_inserter(students));
/***** at this point, the students vector is empty *****/

If I change operator>> to read from a stringstream instead, it works
as expected.

std::istream& operator>>(std::istream& is, Student& s)
{
std::string name = "";
is >> name;

std::string sGrades = "";
std::getline(is, sGrades);

std::stringstream ss(sGrades);

std::vector<int> grades;
std::copy(std::istream_iterator<int>(ss),
std::istream_iterator<int>(), std::back_inserter(grades));

s.setName(name);
s.setGrades(grades);

return is;
}

So even though the one (and only) call to operator>> read the first
line of the file and assigned the values to the Student parameter, the
students vector got nothing added to it. Why would changing to a
stringstream fix this?
 
J

Jim Langston

Given the following input file:

bob 1 2 3 4 5
mary 2 3 4 5 6 7
susan 3 4 5 6 7 8 9

This code snippet does not read it correctly:

class Student
{
public:
...
private:
std::string _Name;
std::vector<int> _Grades;
};
...
/***** this only gets called once *****/
std::istream& operator>>(std::istream& is, Student& s)
{
string name = "";
is >> name;
std::vector<int> grades;
std::copy(std::istream_iterator<int>(is),
std::istream_iterator<int>(), std::back_inserter(grades));

And what happens after the "5" in the first line is read? It will attempt
to read "mary" and "mary" is not an integer. Puts is into an error state.
/***** at this point, name and the grades vector are correct for
the 'first' line in the input file *****/
s.setName(name);
s.setGrades(grades);
return is;
}
...
std::ifstream fin;
std::vector<Student> students;
std::copy(std::istream_iterator<Student>(fin),
std::istream_iterator<Student>(), std::back_inserter(students));
/***** at this point, the students vector is empty *****/

If I change operator>> to read from a stringstream instead, it works
as expected.

std::istream& operator>>(std::istream& is, Student& s)
{
std::string name = "";
is >> name;

std::string sGrades = "";
std::getline(is, sGrades);

Yes, here you are reading one line which stops at the end of line. So for
the first line sGrades will contain "1 2 3 4 5".
std::stringstream ss(sGrades);

std::vector<int> grades;
std::copy(std::istream_iterator<int>(ss),
std::istream_iterator<int>(), std::back_inserter(grades));

s.setName(name);
s.setGrades(grades);

return is;
}

So even though the one (and only) call to operator>> read the first
line of the file and assigned the values to the Student parameter, the
students vector got nothing added to it. Why would changing to a
stringstream fix this?

As indicated. std::copy is attempting to read the rest of the entire file,
names included. std::getline is attempting to read the rest of the line, no
names included since it stops at the end of line terminator.
 
D

david.crow

As indicated. std::copy is attempting to read the rest of the entire file,
names included. std::getline is attempting to read the rest of the line, no
names included since it stops at the end of line terminator.- Hide quoted text -

Thanks for the clarification, Jim.
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

Forum statistics

Threads
473,992
Messages
2,570,220
Members
46,807
Latest member
ryef

Latest Threads

Top