473,385 Members | 1,492 Online
Bytes | Software Development & Data Engineering Community
Post Job

Home Posts Topics Members FAQ

Join Bytes to post your question to a community of 473,385 software developers and data experts.

How to GET multi-word input from a *file* stream as opposed to a *console* stream?

Hi,

I've a got a little (exercise) program that reads data from a file and
puts it into struct members. I run into trouble when one of the data
pieces is comprised of several words (eg "john doe", with a space in
it).

For console input, cin.getline(var, howMuchIWant) or cin.get() has done
the trick for me in the past. It doesn't seem to work for me nearly so
well with a file stream. I wouldn't have thought cpp regarded
file/console streams as significantly different, so I assume I'm doing
something wrong. What am I doing wrong?

thanks for letting me in on the joke! :)

cdj

======example data txt file========
4
john
2000
seattle
peter
4000
san francisco
paul
100
greenlake
mary
10000
seattle
======works fine without the "san" part though======

====== code ======
/*
declare struct type
open file
get # of structs required from file
dynamically make the array of structs
display them
*/

//This routine works fine when the name and location are one word long
only.
//Need to figure out how to utilize inputFile.get or inputFile.getline
to
//read multiword names, locations, and the like.

#include <iostream>
#include <fstream>
#include <cstdlib>

using namespace std;

const int STRSIZE = 60;
const int FILESIZEMAX = 1000;

int testfunction(int arg);

struct donors {
char name[STRSIZE];
double amount;
char location[STRSIZE];
};

int main()
{
char filename[STRSIZE];
ifstream inputFile;

cout << "File name: ";
cin.getline(filename,STRSIZE);

inputFile.open(filename);

if (!inputFile.is_open())
{
cout << "Couldn't open " << filename << endl;
cout << "Terminating execution\n";
exit(EXIT_FAILURE);
}

cout << filename << " successfully opened." << endl;

int numDonors;
inputFile >> numDonors;

cout << "Number of donors in " << filename << ": " << numDonors <<
endl << endl;

if (numDonors==0)
{
cout << "Exiting from \'no donors\' door.\n\n";
exit(EXIT_FAILURE);//Apparently no donor data to read
}

donors * myDonors = new donors[numDonors];
//Now I've allocated structured space for my data
cout << "Donors struct array created with " << numDonors << "
elements." << endl << endl;

//cout << "Name: ";
//cin.getline(myDonors[0].name,STRSIZE);
//cout << "You entered: " << myDonors[0].name << endl; - works fine
for console

for (int i=0; i<numDonors; i++)
{
cout << "Reading record " << i+1 << "... ";

//inputFile.getline(myDonors[i].name,STRSIZE);
//This works for getting multiword input from cin,
//why not for my inputFile object?

inputFile >> myDonors[i].name;
inputFile >> myDonors[i].amount;
inputFile >> myDonors[i].location;
//These work fine for single word items
//Doesn't work for multiword items

cout << "done!" << endl;
}
cout << endl;

for (int i=0; i<numDonors; i++)
{
cout << "Record #" << i+1 << ":" << endl;
cout << "Name: " << myDonors[i].name << endl;
cout << "Amount: " << myDonors[i].amount << endl;
cout << "Location: " << myDonors[i].location << endl << endl;
}

//Close file when done with it.
inputFile.close();

//Free allocated space when done with it.
delete [] myDonors;
return 0;
}

Apr 24 '06 #1
9 2396
sh*************@gmail.com wrote:
Hi,

I've a got a little (exercise) program that reads data from a file and
puts it into struct members. I run into trouble when one of the data
pieces is comprised of several words (eg "john doe", with a space in
it).

For console input, cin.getline(var, howMuchIWant) or cin.get() has done
the trick for me in the past. It doesn't seem to work for me nearly so
well with a file stream. I wouldn't have thought cpp regarded
file/console streams as significantly different, so I assume I'm doing
something wrong. What am I doing wrong?

thanks for letting me in on the joke! :)

cdj

======example data txt file========
4
john
2000
seattle
peter
4000
san francisco
paul
100
greenlake
mary
10000
seattle
======works fine without the "san" part though======

====== code ======
/*
declare struct type
open file
get # of structs required from file
dynamically make the array of structs
display them
*/

//This routine works fine when the name and location are one word long
only.
//Need to figure out how to utilize inputFile.get or inputFile.getline
to
//read multiword names, locations, and the like.

#include <iostream>
#include <fstream>
#include <cstdlib>

using namespace std;

const int STRSIZE = 60;
const int FILESIZEMAX = 1000;

int testfunction(int arg);

struct donors {
char name[STRSIZE];
double amount;
char location[STRSIZE];
};

int main()
{
char filename[STRSIZE];
ifstream inputFile;

cout << "File name: ";
cin.getline(filename,STRSIZE);

inputFile.open(filename);

if (!inputFile.is_open())
{
cout << "Couldn't open " << filename << endl;
cout << "Terminating execution\n";
exit(EXIT_FAILURE);
}

cout << filename << " successfully opened." << endl;

int numDonors;
inputFile >> numDonors;

cout << "Number of donors in " << filename << ": " << numDonors <<
endl << endl;

if (numDonors==0)
{
cout << "Exiting from \'no donors\' door.\n\n";
exit(EXIT_FAILURE);//Apparently no donor data to read
}

donors * myDonors = new donors[numDonors];
//Now I've allocated structured space for my data
cout << "Donors struct array created with " << numDonors << "
elements." << endl << endl;

//cout << "Name: ";
//cin.getline(myDonors[0].name,STRSIZE);
//cout << "You entered: " << myDonors[0].name << endl; - works fine
for console

for (int i=0; i<numDonors; i++)
{
cout << "Reading record " << i+1 << "... ";

//inputFile.getline(myDonors[i].name,STRSIZE);
//This works for getting multiword input from cin,
//why not for my inputFile object?
What, specifically, are you seeing that doesn't work?

inputFile >> myDonors[i].name;
inputFile >> myDonors[i].amount;
inputFile >> myDonors[i].location;
//These work fine for single word items
//Doesn't work for multiword items

cout << "done!" << endl;
}
cout << endl;

for (int i=0; i<numDonors; i++)
{
cout << "Record #" << i+1 << ":" << endl;
cout << "Name: " << myDonors[i].name << endl;
cout << "Amount: " << myDonors[i].amount << endl;
cout << "Location: " << myDonors[i].location << endl << endl;
}

//Close file when done with it.
inputFile.close();

//Free allocated space when done with it.
delete [] myDonors;
return 0;
}


You might also want to consider using std::string instead of fixed
arrays, and then using the nonmember function std::getline() to read
strings.

You also don't deal with the possibility of corrupted data in the first
line of your file (# of donors) -- what happens if you have a negative
value there?
Apr 24 '06 #2
Sorry - here's two outputs, depending on whether the data-piece is
"francisco" or "san francisco". To my limited knowledge, the space in
"san francisco" result in the "inputFile >>" assignment being "thrown
off" by one, yielding screwy results for the stream inputs thereafter.

And yah - there's not much by the way of validation/error handling. It
seems like "covering all the bases" in that respect would take a fair
bit of coding - I just want to get the basic piece functioning
correctly first.

====== good output w/"good" data ======
====== ie, no spaces in the data ======
File name: c:\cdj.txt
c:\cdj.txt successfully opened.
Number of donors in c:\cdj.txt: 4

Donors struct array created with 4 elements.

Reading record 1... done!
Reading record 2... done!
Reading record 3... done!
Reading record 4... done!

Record #1:
Name: john
Amount: 2000
Location: seattle

Record #2:
Name: peter
Amount: 4000
Location: francisco

Record #3:
Name: paul
Amount: 100
Location: greenlake

Record #4:
Name: mary
Amount: 10000
Location: seattle
========= end good output =======

======= bad output =======
======= ie, output when the data has spaces ======
File name: c:\cdj.txt
c:\cdj.txt successfully opened.
Number of donors in c:\cdj.txt: 4

Donors struct array created with 4 elements.

Reading record 1... done!
Reading record 2... done!
Reading record 3... done!
Reading record 4... done!

Record #1:
Name: john
Amount: 2000
Location: seattle

Record #2:
Name: peter
Amount: 4000
Location: san

Record #3:
Name: francisco
Amount: -6.27744e+066
Location:

Record #4:
Name:
Amount: -6.27744e+066
Location:

Press any key to continue . . .
====== end bad output =========

Apr 24 '06 #3
On 24 Apr 2006 12:35:48 -0700 sh*************@gmail.com waved a wand
and this message magically appeared:
Hi,

I've a got a little (exercise) program that reads data from a file and
puts it into struct members. I run into trouble when one of the data
pieces is comprised of several words (eg "john doe", with a space in
it).

For console input, cin.getline(var, howMuchIWant) or cin.get() has
done the trick for me in the past. It doesn't seem to work for me
nearly so well with a file stream. I wouldn't have thought cpp
regarded file/console streams as significantly different, so I assume
I'm doing something wrong. What am I doing wrong?

thanks for letting me in on the joke! :)


Look up token(), it'll help with reading words.

--
http://www.munted.org.uk

Take a nap, it saves lives.
Apr 24 '06 #4
sh*************@gmail.com wrote:
Sorry - here's two outputs, depending on whether the data-piece is
"francisco" or "san francisco". To my limited knowledge, the space in
"san francisco" result in the "inputFile >>" assignment being "thrown
off" by one, yielding screwy results for the stream inputs thereafter.

And yah - there's not much by the way of validation/error handling. It
seems like "covering all the bases" in that respect would take a fair
bit of coding - I just want to get the basic piece functioning
correctly first.

[output redacted]

At the risk of asking the obvious, are you sure you're using getline?
The code you posted uses operator>>, which is whitespace sensitive.
Apr 24 '06 #5
In message <20********************************@munted.org.uk> , Alex
Buell <al********@munted.org.uk> writes
On 24 Apr 2006 12:35:48 -0700 sh*************@gmail.com waved a wand
and this message magically appeared:
Hi,

I've a got a little (exercise) program that reads data from a file and
puts it into struct members. I run into trouble when one of the data
pieces is comprised of several words (eg "john doe", with a space in
it).

For console input, cin.getline(var, howMuchIWant) or cin.get() has
done the trick for me in the past. It doesn't seem to work for me
nearly so well with a file stream. I wouldn't have thought cpp
regarded file/console streams as significantly different, so I assume
I'm doing something wrong. What am I doing wrong?

thanks for letting me in on the joke! :)


Look up token(), it'll help with reading words.


Where? I can't find it in ISO14882.
--
Richard Herring
Apr 26 '06 #6
On 24 Apr 2006 12:35:48 -0700, sh*************@gmail.com wrote:
Hi,

I've a got a little (exercise) program that reads data from a file and
puts it into struct members. I run into trouble when one of the data
pieces is comprised of several words (eg "john doe", with a space in
it).

For console input, cin.getline(var, howMuchIWant) or cin.get() has done
the trick for me in the past. It doesn't seem to work for me nearly so
well with a file stream. I wouldn't have thought cpp regarded
file/console streams as significantly different, so I assume I'm doing
something wrong. What am I doing wrong?

thanks for letting me in on the joke! :)

cdj

======example data txt file========
4
john
2000
seattle
peter
4000
san francisco
paul
100
greenlake
mary
10000
seattle
======works fine without the "san" part though======

====== code ======
/*
declare struct type
open file
get # of structs required from file
dynamically make the array of structs
display them
*/

//This routine works fine when the name and location are one word long
only.
//Need to figure out how to utilize inputFile.get or inputFile.getline
to
//read multiword names, locations, and the like.

#include <iostream>
#include <fstream>
#include <cstdlib>

using namespace std;

const int STRSIZE = 60;
const int FILESIZEMAX = 1000;

int testfunction(int arg);

struct donors {
char name[STRSIZE];
double amount;
char location[STRSIZE];
};

int main()
{
char filename[STRSIZE];
ifstream inputFile;

cout << "File name: ";
cin.getline(filename,STRSIZE);

inputFile.open(filename);

if (!inputFile.is_open())
{
cout << "Couldn't open " << filename << endl;
cout << "Terminating execution\n";
exit(EXIT_FAILURE);
}

cout << filename << " successfully opened." << endl;

int numDonors;
inputFile >> numDonors;
This leaves the newline character. use inputFile.get() after this call
to extract and throw away the newline character.

cout << "Number of donors in " << filename << ": " << numDonors <<
endl << endl;

if (numDonors==0)
{
cout << "Exiting from \'no donors\' door.\n\n";
exit(EXIT_FAILURE);//Apparently no donor data to read
}

donors * myDonors = new donors[numDonors];
//Now I've allocated structured space for my data
cout << "Donors struct array created with " << numDonors << "
elements." << endl << endl;

//cout << "Name: ";
//cin.getline(myDonors[0].name,STRSIZE);
//cout << "You entered: " << myDonors[0].name << endl; - works fine
for console

for (int i=0; i<numDonors; i++)
{
cout << "Reading record " << i+1 << "... ";

//inputFile.getline(myDonors[i].name,STRSIZE);
//This works for getting multiword input from cin,
//why not for my inputFile object?
After putting the inputFile.get() after the extraction operator the
above getline should work.
inputFile >> myDonors[i].name;
inputFile >> myDonors[i].amount; This wil also leave the newline character. use inputFile.get() after
this call to extract and throw away the newline character.
and then use the getline() function to get the location.
inputFile >> myDonors[i].location;
//These work fine for single word items
//Doesn't work for multiword items

cout << "done!" << endl;
}
cout << endl;

for (int i=0; i<numDonors; i++)
{
cout << "Record #" << i+1 << ":" << endl;
cout << "Name: " << myDonors[i].name << endl;
cout << "Amount: " << myDonors[i].amount << endl;
cout << "Location: " << myDonors[i].location << endl << endl;
}

//Close file when done with it.
inputFile.close();

//Free allocated space when done with it.
delete [] myDonors;
return 0;
}


The extraction operator ( >> ) does not remove the newline character
from the stream so the next call to getline() will only get the
newline character. To remove the newline character use inputFile.get()
after using the extraction operator (inputFile >> ).
Apr 26 '06 #7
On Wed, 26 Apr 2006 17:56:18 +0100 Richard Herring <ju**@[127.0.0.1]>
waved a wand and this message magically appeared:
thanks for letting me in on the joke! :)


Look up token(), it'll help with reading words.


Where? I can't find it in ISO14882.


Ok well tokenising is a trick I use to parse sentences.

std::string input("hello world!");
std::stringstream ss(input);
std::vector<string> tokens;
std::string buffer;

while (ss >> buffer)
tokens.push_back(buffer);

Now you get a nice little vector<std::string> full of words. In this
case; tokens.at(0) has "hello", and tokens.at(1) has "world!".

Capise?

--
http://www.munted.org.uk

Take a nap, it saves lives.
Apr 26 '06 #8
In message <20********************************@munted.org.uk> , Alex
Buell <al********@munted.org.uk> writes
On Wed, 26 Apr 2006 17:56:18 +0100 Richard Herring <ju**@[127.0.0.1]>
waved a wand and this message magically appeared:
>> thanks for letting me in on the joke! :)
>
>Look up token(), it'll help with reading words.


Where? I can't find it in ISO14882.


Ok well tokenising is a trick I use to parse sentences.

std::string input("hello world!");
std::stringstream ss(input);
std::vector<string> tokens;
std::string buffer;

while (ss >> buffer)
tokens.push_back(buffer);

Now you get a nice little vector<std::string> full of words. In this
case; tokens.at(0) has "hello", and tokens.at(1) has "world!".

Capise?


Capisco, though I see nothing called token() above ;-).

But if the input is a big file, you may get a nice _big_ vector filling
most of your memory with strings.

That's an unnecessary overhead if the intention is to scan the strings
sequentially and then discard them. This is what istream_iterators are
for.

std::string input("hello world");
std::stringstream ss(input);
for (std::istream_iterator<std::string, char> p(ss), e; p!=e; ++p)
{
// do stuff with *p
}

The above also uses a rather simplistic definition of "word" - if your
text has punctuation, you probably need something more complex. Time to
start looking at boost::tokenizer, or even the Spirit parser.

--
Richard Herring
Apr 27 '06 #9
On Thu, 27 Apr 2006 10:25:59 +0100 Richard Herring <ju**@[127.0.0.1]>
waved a wand and this message magically appeared:
Capisco, though I see nothing called token() above ;-).

But if the input is a big file, you may get a nice _big_ vector
filling most of your memory with strings.

That's an unnecessary overhead if the intention is to scan the
strings sequentially and then discard them. This is what
istream_iterators are for.

std::string input("hello world");
std::stringstream ss(input);
for (std::istream_iterator<std::string, char> p(ss), e; p!=e; ++p)
{
// do stuff with *p
}


Oh yes, you're right my code is only good for small strings. Thanks for
the above, I might find it useful one day!

--
http://www.munted.org.uk

Take a nap, it saves lives.
Apr 27 '06 #10

This thread has been closed and replies have been disabled. Please start a new discussion.

Similar topics

4
by: OutsiderJustice | last post by:
Hi All, I can not find any information if PHP support multi-thread (Posix thread) or not at all, can someone give out some information? Is it supported? If yes, where's the info? If no, is it...
37
by: ajikoe | last post by:
Hello, Is anyone has experiance in running python code to run multi thread parallel in multi processor. Is it possible ? Can python manage which cpu shoud do every thread? Sincerely Yours,...
12
by: * ProteanThread * | last post by:
but depends upon the clique: ...
6
by: Joe | last post by:
I have 2 multi-list boxes, 1 displays course categories based on a table called CATEGORIES. This table has 2 fields CATEGORY_ID, CATEGORY_NAME The other multi-list box displays courses based on...
4
by: mimmo | last post by:
Hi! I should convert the accented letters of a string in the correspondent letters not accented. But when I compile with -Wall it give me: warning: multi-character character constant Do the...
5
by: dkelly925 | last post by:
Is there a way to add an If Statement to the following code so if data in a field equals "x" it will launch one report and if it equals "y" it would open another report. Anyone know how to modify...
17
by: =?Utf-8?B?R2Vvcmdl?= | last post by:
Hello everyone, Wide character and multi-byte character are two popular encoding schemes on Windows. And wide character is using unicode encoding scheme. But each time I feel confused when...
0
by: Sabri.Pllana | last post by:
We apologize if you receive multiple copies of this call for papers. *********************************************************************** 2008 International Workshop on Multi-Core Computing...
1
by: mknoll217 | last post by:
I am recieving this error from my code: The multi-part identifier "PAR.UniqueID" could not be bound. The multi-part identifier "Salary.UniqueID" could not be bound. The multi-part identifier...
14
by: =?ISO-8859-1?Q?Tom=E1s_=D3_h=C9ilidhe?= | last post by:
As far as I know, the C Standard has no mention of multi-threaded programming; it has no mention of how to achieve multi-threaded programming, nor does it mention whether the language or its...
1
by: CloudSolutions | last post by:
Introduction: For many beginners and individual users, requiring a credit card and email registration may pose a barrier when starting to use cloud servers. However, some cloud server providers now...
0
by: ryjfgjl | last post by:
In our work, we often need to import Excel data into databases (such as MySQL, SQL Server, Oracle) for data analysis and processing. Usually, we use database tools like Navicat or the Excel import...
0
by: taylorcarr | last post by:
A Canon printer is a smart device known for being advanced, efficient, and reliable. It is designed for home, office, and hybrid workspace use and can also be used for a variety of purposes. However,...
0
by: Charles Arthur | last post by:
How do i turn on java script on a villaon, callus and itel keypad mobile phone
0
by: aa123db | last post by:
Variable and constants Use var or let for variables and const fror constants. Var foo ='bar'; Let foo ='bar';const baz ='bar'; Functions function $name$ ($parameters$) { } ...
0
by: ryjfgjl | last post by:
If we have dozens or hundreds of excel to import into the database, if we use the excel import function provided by database editors such as navicat, it will be extremely tedious and time-consuming...
0
by: emmanuelkatto | last post by:
Hi All, I am Emmanuel katto from Uganda. I want to ask what challenges you've faced while migrating a website to cloud. Please let me know. Thanks! Emmanuel
0
BarryA
by: BarryA | last post by:
What are the essential steps and strategies outlined in the Data Structures and Algorithms (DSA) roadmap for aspiring data scientists? How can individuals effectively utilize this roadmap to progress...
1
by: nemocccc | last post by:
hello, everyone, I want to develop a software for my android phone for daily needs, any suggestions?

By using Bytes.com and it's services, you agree to our Privacy Policy and Terms of Use.

To disable or enable advertisements and analytics tracking please visit the manage ads & tracking page.