Функция scanf
и другие члены семейства, как правило, не высоко ценятся в основном по трем причинам:
□ традиционно их реализации полны ошибок;
□ в использовании эти функции не гибки;
□они могут привести к созданию программного кода, в котором трудно решить, что подвергать синтаксическому анализу.
В качестве альтернативы попытайтесь применять другие функции, такие как fread
или fgets
, для чтения строк ввода, а затем воспользуйтесь строковыми функциями для разделения введенной строки на нужные элементы.
В библиотеке stdio существует ряд других функций, использующих потоки как параметры или стандартные потоки stdin
, stdout
, stderr
:
□ fgetpos
— возвращает текущую позицию в файловом протоке;
□ fsetpos
— устанавливает текущую позицию в файловом потоке;
□ ftell
— возвращает величину текущего смещения файла в потоке;
□ rewind
— сбрасывает текущую позицию файла в потоке и переводит ее в начало файла;
□ freopen
— повторно использует файловый поток;
□ setvbuf
— задает схему буферизации для потока;
□ remove
— эквивалент функции unlink
, до тех пор пока параметр path
не является каталогом, в этом случае она эквивалентна функции rmdir
.
Эти библиотечные функции описаны на страницах интерактивного справочного руководства в разделе 3.
Вы можете использовать функции обработки файловых потоков для повторной реализации с их помощью программы копирования файлов. Взгляните на программу copy_stdio.c в упражнении 3.3.
Упражнение 3.3. Третья версия программы копирования файлов
Эта программа очень похожа на предыдущие версии, но посимвольное копирование выполняется с помощью вызовов функций, заданных в файле stdio.h:
#include
#include
int main() {
int c;
FILE *in, *out;
in = fopen("file.in", "r");
out = fopen("file.out", "w");
while((c = fgetc(in)) != EOF) fputc(c, out);
exit(0);
}
Выполнив эту программу, как прежде, вы получите:
$ TIMEFORMAT="" time ./copy_stdio
0.06user 0.02system 0:00.11elapsed 81%CPU
Как это работает
На этот раз программа выполняется 0,11 с, не так быстро, как низкоуровневая блочная версия, но значительно быстрее другой посимвольной версии. Это произошло потому, что библиотека stdio поддерживает внутренний буфер в структуре FILE
, и низкоуровневые системные вызовы выполняются, только когда буфер заполняется. Не ленитесь экспериментировать, тестируя программный код построчного и блочного копирования с помощью stdio, чтобы увидеть, как они действуют в случае проверенных нами трех примеров.
Для обозначения ошибок многие функции библиотеки stdio применяют значения за пределами допустимых, например, пустые указатели или константу EOF
. В этих случаях ошибка указывается во внешней переменной errno
.
#include
extern int errno;
Примечание
Имейте в виду, что многие функции могут изменять значение errno
. Оно достоверно, только когда функция закончилась неудачно. Вам следует проверять это значение сразу же, как функция сообщила о сбое. Прежде чем использовать его, скопируйте это значение в другую переменную, поскольку функции вывода, такие как fprintf
, могут сами изменять errno
.
Вы можете также запросить состояние файлового потока, чтобы выяснить, возникла ли ошибка или достигнут конец файла.
#include
int ferror(FILE *stream);
int feof(FILE *stream);
void clearerr(FILE *stream);
Функция ferror
проверяет индикатор ошибок потока и возвращает ненулевое значение, если индикатор установлен, и ноль в противном случае.
Функция feof
проверяет индикатор конца файла в потоке и возвращает ненулевое значение, если индикатор установлен, или ноль в противном случае. Применяйте ее следующим образом:
if (feof(some_stream))
/* Мы в конце */
Функция clearerr
очищает индикаторы конца файла и ошибки для потока, на который указывает параметр stream
. Она не возвращает никакого значения, и для нее не определены никакие ошибки. Вы можете применять эту функцию для сброса состояния ошибки в потоках. Примером может быть возобновление записи в поток после разрешения проблемы, связанной с ошибкой "disk full" (диск заполнен).
Потоки и дескрипторы файлов
Читать дальше