Character level I/O
C provides (through its libraries) a variety of I/O routines. At the character level, getchar() reads one character at a time from stdin, while putchar() writes one character at a time to stdout. For example, consider
#include <>
void main()
{
int i, nc;
nc = 0;
i = getchar();
while (i != EOF) {
nc = nc + 1;
i = getchar();
}
printf("Number of characters in file = %d\n", nc);
}
This program counts the number of characters in the input stream (e.g. in a file piped into it at execution time). The code reads characters (whatever they may be) from stdin (the keyboard), uses stdout (the X-terminal you run from) for output, and writes error messages to stderr (usually also your X-terminal). These streams are always defined at run time. EOF is a special return value, defined in stdio.h, returned by getchar() when it encounters an end-of-file marker when reading. Its value is computer dependent, but the C compiler hides this fact from the user by defining the variable EOF. Thus the program reads characters from stdin and keeps adding to the counter nc, until it encounters the ``end of file''.
An experienced C programmer would probably code this example as:
#include <>
void main()
{
int c, nc = 0;
while ( (c = getchar()) != EOF ) nc++ ;
printf("Number of characters in file = %d\n", nc);
}
C allows great brevity of expression, usually at the expense of readability!
The () in the statement (c = getchar()) says to execute the call to getchar() and assign the result to c before comparing it to EOF; the brackets are necessary here. Recall that
The UNIX command wc counts the characters, words and lines in a file. The program above can be considered as your own wc. Let's add a counter for the lines.
#include <>
void main()
{
int c, nc = 0, nl = 0;
while ( (c = getchar()) != EOF )
{
nc++ ;
if (c == '\n') nl++;
}
printf("Number of characters = %d, number of lines = %d\n",
nc, nl);
}
Can you think of a way to count the number of words in the file?
Higher-Level I/O capabilities
We have already seen that printf handles formatted output to stdout. The counterpart statement for reading from stdin is scanf. The syntax
scanf("format string", variables);
resembles that of printf. The format string may contain blanks or tabs (ignored), ordinary ASCII characters, which must match those in stdin, and conversion specifications as in printf.
Equivalent statements exist to read from or write to character strings. They are:
sprintf(string, "format string", variables);
scanf(string, "format string", variables);
The ``string'' argument is the name of (i.e. a pointer to) the character array into which you want to write the information.
I/O to and from files
Similar statements also exist for handling I/O to and from files. The statements are
#include <>
FILE *fp;
fp = fopen(name, mode);
fscanf(fp, "format string", variable list);
fprintf(fp, "format string", variable list);
fclose(fp );
The logic here is that the code must
- define a local ``pointer'' of type FILE (note that the uppercase is necessary here), which is defined in <>
- ``open'' the file and associate it with the local pointer via fopen
- perform the I/O operations using fscanf and fprintf
- disconnect the file from the task with fclose
The ``mode'' argument in the fopen specifies the purpose/positioning in opening the file: ``r'' for reading, ``w'' for writing, and ``a'' for appending to the file. Try the following:
#include <>
void main()
{
FILE *fp;
int i;
fp = fopen("foo.dat", "w"); /* open foo.dat for writing */
fprintf(fp, "\nSample Code\n\n"); /* write some info */
for (i = 1; i <= 10 ; i++)
fprintf(fp, "i = %d\n", i);
fclose(fp); /* close the file */
}
Compile and run this code; then use any editor to read the file foo.dat.