Для работы с потоками существуют стандартные библиотеки stdio, string, stdlib, io, которые становятся доступными из функций пользователя после использования директивы препроцессора #include, например,
#include < stdio.h >
Вызов подсказки для получения списка функций библиотеки:
1) В среде С выполнить команду HELP→Contents и нажать Enter;
2) Выбрать раздел Header Files и нажать Enter;
3) Выбрать название библиотеки и нажать Enter;
4) Выбрать функцию.
1.2. Работа с файлами (потоками)
1.2.1. Этапы работы с файлами (потоками)
Файл в программе на языке С – это переменная–указатель на тип FILE, называемая файловой переменной.
Работа с файлами состоит их трех этапов:
1. Открытие файла;
2. Обработка файла;
3. Закрытие файла.
1.2.2. Открытие файла (потока)
Поток можно открыть для чтения, записи, и для чтения и записи с помощью стандартной функции fopen, прототип которой имеет вид:
FILE *fopen (const char *filename, const char*mode);
Здесь первый параметр функции (filename) – имя открываемого файла в виде строки символов, второй (mode) - режим открытия файла (тип доступа к файлу)– строка из одного и более символов:
" r " – файл открывается для чтения;
" w " – открывается пустой файл для записи (если файл существует, то его содержимое теряется);
" a " – файл открывается для добавления информации в конец файла; если файла нет, он создается;
" r+ " – файл открывается для чтения и записи (файл должен существовать);
" w+ " – открывается пустой файл для записи и чтения (если файл существует, то его содержимое теряется);
" a+ " – файл открывается для чтения и добавления данных в конец файла.
" t " – открытие файла в текстовом режиме;
" b " – открытие файла в двоичном режиме.
|
Примечание.
По умолчанию (когда в режиме открытия файла отсутствует t или b) файл открывается в текстовом режиме.
Возможны следующие режимы доступа: "w+b", "wb", "rw+", "w+t", "rt+" и др.
При успешном открытии потока функция fopen возвращает указатель на структуру типа FILE. Эта структура связана с физическим файлом и содержит всю необходимую информацию для работы с ним (указатель на текущую позицию в файле, тип доступа и др.).
Возвращаемое функцией значение нужно сохранить и использовать для ссылки на открытый файл.
Если произошла ошибка при открытии файла, то возвращается NULL.
Т.о., для работы с функцией fopen в программе на языке С необходимо:
1. Объявить переменную-указатель на структуру типа FILE, например:
FILE* f;
Эта переменная (f) называется файловой переменной или файлом;
2. Файловой переменной присвоить значение, возвращаемое функцией fopen, например:
f=fopen("c:\mdoc\d.txt", "r+");
Примечание. Описание и инициализация файловой переменной (например, f) может быть осуществлено в одном операторе, например:
FILE*f = fopen ("c:\mdoc\d.txt","r+");
После открытия файла все действия над файловой переменной будут отождествляться с действиями над внешним файлом (физическим файлом), т.е. с файловой переменной связывается определенный физический поток. При этом структуре типа FILE выделяется область ОП, адрес которой возвращает функция fopen. Поток связывается со структурой типа FILE.
Структура типа FILE используется функциями ввода/вывода для хранения информации, связанной с устройством или файлом. Указатель на тип FILE используется для операций с файлом с помощью библиотечных функций. Его передают библиотечным функциям в качестве параметра. Такие библиотечные функции называются функциями ввода/вывода.
|
Примечания.
1. При открытии потока с ним связывается область памяти – буфер.
2. При аварийном завершении программы выходной буфер может быть не выгружен, поэтому возможна потеря данных.
3. С помощью функций setbuf и setvbuf можно управлять размерами и наличием буферов.
4. Перед началом работы программы операционная система автоматически открывает пять стандартных потоков:
1) stdin – стандартный ввод;
2) stdout – стандартный вывод;
3) stdprn – стандартная печать;
4) stderr – стандартный вывод сообщений об ошибках;
5) stdaux – стандартный дополнительный поток (стандартные порты).
Потоки 1), 2), 4) относятся к консоли (con), 3) – к prn, 5) – к aux (коммуникационный канал), т.е., в начале работы программы автоматически открываются потоки следующими операторами:
1) FILE *stdin=fopen (" con ", " r ");
2) FILE *stdout=fopen (" con ", " w ");
3) FILE *stdprn=fopen("prn", "w");
4) FILE *stderr=fopen (" con ", " w ");
5) FILE *stdaux=fopen (" aux ", " wb ");
Файлы stdin и stdout можно переназначать при запуске exe-программы или в программе с помощью функции freopen. Ее прототип:
File * freopen (const char *filename, const char *mode, FILE *stream);
где freopen – имя функции;
filename – имя файла;
mode – режим работы файла;
stream - поток.
Функция freopen работает аналогично функции fopen, но предварительно закрывает поток stream, если тот был ранее открыт.
1.2.2. Закрытие файла (потока)
Для закрытия потоков, используемых в программе, применяются стандартные функции fclose и fcloseall.
|
Один заданный поток закрывается функцией fclose. Ее прототип:
int fclose (FILE *stream);
где stream – поток.
Пример обращения к функции fclose:
fclose (f);
Если поток успешно закрыт, функция fclose возвращает значение 0. Если при закрытии потока произошла ошибка – значение EOF (-1).
Функция fcloseall закрывает все потоки, открытые с помощью fopen, кроме stdin, stdout, stderr. Прототип функции fcloseall:
int fcloseall ();
Функция fcloseall возвращает количество закрытых потоков или значение EOF (-1), если при закрытии произошла ошибка.
Примечание. Функция perror позволяет вывести сообщение об ошибке при неуспешном закрытии файла. Ее прототип:
void perror (const char *s);
Пример.
FILE * fp;
fp=fopen("d.dat", "r");
if(!fp) perror("Нельзя открыть файл для чтения");
else fclose ("d.dat");
1.2.3. Удаление файла
Удалить файл можно с помощью функции remove. Ее прототип:
int remove (const char *filename);
где filename – указатель на строку с именем физического файла (имя файла).
Функция remove возвращает значение 0 при успешном удалении файла и не нуль в противном случае. Открытый файл необходимо предварительно закрыть.
Пример открытия и закрытия файла.
В текстовый файл с именем rez.txt записываются результаты выполнения программы в начало нового файла, если файл не существует, или в конец существующего файла.
#include <stdio.h>
void main ()
{
int n; //номер теста
float y,x;
FILE *f; //Описание файловой переменной f (файла)
puts(“Введите x”);
scanf(“%f”,&x);
y=2*x;
f=fopen("rez.txt", "a"); //Открытие файла
puts("Номер теста?");
scanf("%d",&n);
fprintf(f,"Тест N %d\n",n); //Запись в файл f значения
fprintf(f,"%f %f\n",x,y);
fclose(f); //Закрытие файла
fflush(stdin); getchar();
}
В результате первого выполнения программы в текущем каталоге будет создан файл с именем rez.txt. В его начало запишутся значения n, x, y. При последующих запусках программы на выполнение значения n, x, y будут записываться в конец файла rez.txt, на что указывает режим " a " открытия файла в функции fopen. Отсутствие символа " t " или " b " в режиме открытия файла "говорит" о работе с текстовым файлом (но не расширение txt в имени файла).
1.3. Ввод/вывод в поток
1.3.1. Основные понятия
Ввод/вывод в поток можно осуществить разными способами:
· в виде последовательности байтов;
· в виде символов и строк;
· с использованием форматных преобразований.
Операции ввода/вывода выполняются, начиная с текущей позиции потока. Текущая позиция определяется положением указателя потока. Указатель потока устанавливается на начало или конец файла при открытии в соответствии с режимом открытия. После каждой операции ввода/вывода указатель потока автоматически изменяется.
1.3.2. Позиционирование в файле
Функции получения текущего положения указателя потока
ftell и fgetpos
Прототип функции ftell (из <stdio.h>):
long int ftell(File *f);
Функция ftell возвращает текущую позицию в файле, связанном с потоком f, как длинное целое. В случае ошибки возвращает число (-1).
Прототип функции fgetpos (из <stdio.h>):
int fgetpos (File *f, fpos_t *pos);
Функция fgetpos возвращает текущую позицию в файле, связанном с потоком f, и копирует значение текущей позиции по адресу pos. Это значение позднее может использоваться функцией. Возвращаемое значение имеет тип fpos_t, который используется функциями fgetpos и fsetpos для хранения текущей позиции файла:
typedef long fpos_t;
1.3.3. Функции задания положения указателя fseek и fsetpos
Прототип функции fseek:
int fseek (File *f, long off, int org);
Функция перемещает текущую позицию в файле, связанном с потоком f, на позицию off, отсчитываемую от значения org, которое должно быть равно одной из констант, определенных в < stdio.h >:
SEEK_CUR – от текущей позиции указателя (1);
SEEK_END – от конца файла (2);
SEEK_SET - от начала файла (0).
Параметр off задает количество байтов, на которое необходимо сместить указатель соответственно параметру org. Величина смещения может быть как положительной, так и отрицательной, но нельзя сместиться за пределы начала файла. Такой доступ к данным в файле называется произвольным.
Функция возвращает …
Прототип функции fsetpos:
int fsetpos (File *f, const fops_t *pos);
Функция перемещает текущую позицию в файле, связанном с потоком f, на позицию *pos, предварительно полученную с помощью функции fgetpos.
Примечание. Функции fseek и fsetpos нельзя использовать для стандартных потоков.
Функция rewind очищает флаги ошибок при работе с потоками и переходит к началу файла. Ее прототип:
void rewind(File *f);
1.3.4.Функции чтения и записи потока байтов fread и fwrite
Прототип функции fread:
size_t fread (void *buffer, size_t size, size_t count, FILE *stream);
Функция считывает count элементов длиной size байтов в область, заданную указателем buffer, из потока stream. Возвращает количество прочитанных элементов, которое может быть меньше count, если при чтении произошла ошибка или встретился конец файла.
Прототип функции fwrite:
size_t fwrite (const void *p, size_t size, size_t, FILE *);
Функция записывает n элементов длиной size байтов из буфера, заданного указателем p, в поток f. Возвращает число записанных элементов.
1.3.5. Функции чтения символа из потока (getс, fgetс, getchar)
Функция getс, имеющая прототип:
int getc (FILE *f);
возвращает очередной символ в формате int из потока f. Если символ не может быть прочитан, то возвращает значение EOF.
Функция fgetс, имеющая прототип:
int fgetc (FILE *f);
возвращает очередной символ в формате int из потока f. Если символ не может быть прочитан, то возвращает значение EOF.
Функция getchar, имеющая прототип:
int getchar (void);
возвращает очередной символ в формате int из стандартного потока (stdin). Если символ не может быть прочитан, то возвращает значение EOF.
1.3.6. Функции записи символа в поток (putc, fputc, putchar)
Функция puts записывает символ ch в поток f. При ошибке возвращает значение EOF. Ее прототип:
int putc (int ch, FILE *f);
Выполняется аналогично функция fputc. Ее прототип:
int fputc (int ch, FILE *f);
Функция putchar выводит символ ch на стандартное устройство вывода, добавляя в конце символ новой строки. Возвращает неотрицательноу значение при успехе или EOF – при ошибке.
1.3.7. Функции чтения строки из потока (fgets, gets)
Прототип функции fgets:
char *fgets (char *s, int n, FILE *f);
Функция читает не более n-1 байт из потока f в строку s, прекращая чтение при обнаружении символа новой строки или конца файла. Символ новой строки при чтении не отбрасывается, помещается конец строки s. Прочитанная строка дополняется ограничителем строки ('\0'). При обнаружении ошибки или конца файла возвращает NULL, в противном случае – указатель на строку s.
Функция читает символы с клавиатуры до появления символа новой строки и помещает их в строку s. Сам символ новой строки в строку не включается. Возвращает указатель на s. Прототип функции gets:
char *gets (char *s);
1.3.8. Функции записи строки в поток (fputs, puts)
Прототип функции fputs:
int fputs (const char * s, FILE *f);
Функция fputs записывает строку символов s в поток f. Символ конца строки в файл не записывается. При ошибке возвращает значение EOF, иначе – неотрицательное число.
Функция puts выводит строку s на стандартное устройство вывода, добавляя в конце символ новой строки. Возвращает неотрицательное значение при успехе или EOF – при ошибке. Прототип функции puts:
int puts (char *s);
1.3.9. Функции форматированного ввода из потока (, scanf, sscanf)
Прототип функции fscanf:
intfscanf (FILE *f, const char *fmt [, par1, par2, …]);
Эта функция вводит (читает) строку параметров par1, par2, … в формате, определенном строкой fmt, из файла f. Возвращает число переменных, которым присвоено значение.
Прототип функции scanf:
intscanf (const char *fmt [, par1, par2, …]);
Функция scanf вводит (читает) строку параметров par1, par2, … в формате, определенном строкой fmt, со стандартного устройства ввода (обычно с клавиатуры (stdin)). Возвращает число переменных, которым присвоено значение.
Прототип функции sscanf:
intsscanf (const char *buf, const char *fmt [, par1, par2, …]);
Функция sscanf вводит данные аналогично функции scanf, но не с клавиатуры, а из строки символов, переданной ей первым параметром. Аргумент buf – строка символов, из которой вводятся значения, fmt - строка формата, в соответствии с которой происходит преобразование данных. Многоточие указывает на наличие необязательных аргументов, соответствующих адресам вводимых значений.
1.3.10. Функции форматированного вывода в поток (fprintf, printf, sprintf)
Прототип функции fprintf:
int fprintf (FILE *f, const char *fmt, …);
Функция записывает в поток f переменные, список которых обозначен многоточием (…), в формате, указанном строкой fmt. Возвращает число записанных значений.
Прототип функции printf:
int printf (const char *fmt, …);
Функция выводит на стандартное устройство вывода (stdout) значенияпеременных, перечисленных в списке, обозначенном многоточием (…), в формате, определенном строкой fmt. Возвращает число выведенных значений.
Прототип функции sprintf:
int sprintf (char *buf, const char *fmt, …);
Функция выводит в строку значения переменных, перечисленных в списке, обозначенном многоточием (…), в формате, определенном строкой fmt.
Примечание. Спецификации формата fmt были рассмотрены в разделе "Консольный ввод-вывод".
1.4. Обработка ошибок
Функции работы с потоками возвращают значения, которые рекомендуется анализировать в программе и обрабатывать ошибочные ситуации, возникающие, например, при открытии файлов или чтении из потока. При работе с файлами часто используют функции feof и ferror.
Прототип функции feof:
int feof (FILE *f);
Функция возвращает EOF или значение, отличное от 0, если достигается конец файла; в противном случае 0.
Прототип функции ferror:
int ferror (FILE *f);
Возвращает не равное нулю целое значение, означающее код ошибки, если обнаружена ошибка ввода/вывода; 0 – в противном случае.
1.5. Пример обработки текстового файла
В текстовом файле "dbase.txt" хранятся данные о сотрудниках фирмы. В каждой строке файла указана фамилия, год рождения, оклад сотрудника. Для простоты обработки файла данные записаны единообразно: первые 15 символов занимает фамилия, следующие 5 – год рождения, последние 10 – оклад.
Требуется, интерпретируя структурами строки файла, вывести на экран или принтер содержимое файла в виде таблицы, создать из строк файла с данными о сотрудниках файла, моложе 20 лет, массив структур. Вывести на экран или в текстовый файл полученный массив или вывести сообщение о том, что такие молодые сотрудники не работают в фирме.
Структура записи файла изображена на рисунке.
Фамилия | Год рожден. | Оклад |
Записи файла могут иметь вид:
Иванов И.П. 1982 453120
Авчинникова 1999 578320
Васильков 1955 456780
Чернов 1967 1345600
Требуется на экран, на принтер или в текстовый файл вывести список в форме таблицы. Например:
СПИСОК СОТРУДНИКОВ ФИРМЫ
Фамилия | Год рожд. | Оклад |
Иванов И.П. | ||
Авчинникова | ||
Васильков | ||
Чернов |
/* Поиск в массиве структур, читаемых из текстового файла */
#include <stdio.h> // с записью рез-тов в новый текстовый файл
#include <string.h>
#include <stdlib.h>
// #include <windows.h>
main()
{
const int lfio=15, //длина поля фио,
lpay=5, // длина поля г.рожд
lo=10, //длина поля оклада
l=lfio+lpay+lo; //длина записи в файле
struct Man
{ char fio[lfio]; // фио
int year; // год рожд.
float pay; // оклад
};
Man db;
char s[l]; //строка для записи файла
FILE *fin, // Исх. файл
*fo,*f1; // Вых. файлы
if ((fin=fopen("dbase.txt", "r"))==NULL)
{puts("Ошибка открытия файла\n");
fflush(stdin); getchar(); return 1;}
puts("Файл открыт");
fo=fopen("dbout.txt","w");
f1=fopen("dbout1.txt","w");
while (!feof(fin))
{ fgets(s,l,fin); puts(s);
strncpy(db.fio,s,lfio-1);
db.fio[lfio-1]='\0';
db.year=atoi(&s[15]);
db.pay=atof(&s[20]);
/*if(db.year>2000)*/
fprintf(fo,"%-15s% %10.1f\n", db.fio, db.pay);
fputs(s,f1);
}
fclose(fin); fclose(fo); fclose(f1);
fflush(stdin); getchar();
return 0;
}
1.6. Пример обработки текстового и бинарного файла
/* Построчное считывание данных из текстового файла " dbase.txt " в буферную переменную s, формирование из них структуры db и запись ее в двоичном режиме в выходной файл " dbout.dat ".
Считывание из двоичного файла записи с номером i и вывод ее на экран.
Считывание из двоичного файла записей и вывод на экран только тех записей, для которых фамилия есть " ivanoff ". */
Таблица соответствия
№ | Идентификатор | Тип | Комментарий |
lfio | const int | Длина поля фио | |
lpay | const int | Длина поля г. рожд. | |
lo | const int | Длина поля оклада | |
l | const int | Длина записи | |
db | Запись | Запись | |
s | строка | Строка с содерж. записи | |
fin | Текстовый файл | Исходный текстовый файл | |
fo | Двоичный файл | Двоичный файл, получ. из fin | |
kol | int | Кол-во записей файла fin | |
i | int | Номер записи файла fin |
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <iostream.h>
main()
{
const int lfio=15,lpay=5,lo=10,//длины полей: фио, г.рожд., оклада в т. файле
l=lfio+lpay+lo; //длина записи в т. файле
struct Man
{ char fio[lfio]; // фио
int year; // год рожд.
float pay; // оклад
};
Man db;
char s[l]; //строка для записм файла
FILE *fin, // Исх. файл
*fo; // Вых. файлы
if ((fin=fopen("dbase.txt", "r"))==NULL)
{puts("Ошибка открытия файла\n");
fflush(stdin); getchar(); return 1;}
puts("Файл открыт");
fo=fopen("dbout.dat","w+b");
int kol=0;
while (!feof(fin))
{ fgets(s,l,fin); puts(s);
strncpy(db.fio,s,lfio-1);
db.fio[lfio-1]='\0';
db.year=atoi(&s[15]);
db.pay=atof(&s[20]);
fwrite(&db,sizeof db,1,fo);
kol++;
}
fclose(fin);
int i; printf("Введите номер записи (0-%d)",kol-1);
cin>>i;
if(i>=kol) { cout<<"Запись не существует"; return 1;}
fseek(fo,(sizeof db)*i,SEEK_SET);
fread(&db,sizeof db,1,fo);
cout << db.fio << " "<< db.year<<" " << db.pay;
fseek(fo,0,SEEK_SET);
i=0;
while (i<kol)
{ printf("\n");
fread(&db,sizeof db,1,fo);
if(!strcmp(db.fio," ivanoff ")) cout<<"Есть "<<db.fio;
i++;
}
fclose(fo);
fflush(stdin); getchar();
return 0;
}