Stdio h что за библиотека
Stdio h что за библиотека
БлогNot. Лекции по C/C++: работа с файлами (stdio.h)
Лекции по C/C++: работа с файлами (stdio.h)
Параметр имя_файла может содержать относительный или абсолютный путь к открываемому файлу:
3) имя файла запрашивается у пользователя:
Параметр режим_доступа определяет, какие действия будут разрешены с открываемым файлом, примеры его возможных значений:
3) «at» – открываем текстовый файл для добавления данных в конец файла;
Фактически, указание «r» или «t» не накладывает каких-либо ограничений на методы, которые мы будем применять для чтения или записи данных.
После открытия файла следует обязательно проверить, удалась ли эта операция. Для этого есть 2 основных подхода:
1) стандартный обработчик ferror (см. пособиe, п.8.7);
Пример. Приложение проверяет, удалось ли открыть файл из текущей папки, имя файла запрашивается у пользователя (Visual Studio)
2) в начало файла (до всех #include ) включить директиву
Выбор способа чтения или записи данных зависит от того, какой должна быть структура файла.
Пример. Файл text.txt в текущей папке приложения имеет следующий вид:
Прочитаем его как последовательность вещественных чисел.
2. На «восприятие» программой данных может влиять установленная в приложении локаль. Например, если до показанного кода выполнен оператор результат работы кода может измениться (для русской локали разделителем целой и дробной части числа является запятая, а не точка).
3. Очередное чтение данных изменяет внутренний файловый указатель. Этот указатель в любой момент времени, пока файл открыт, показывает на следующее значение, которое будет прочитано. Благодаря этому наш код с «бесконечным» while не зациклился.
4. Код показывает, как читать из файла заранее неизвестное количество значений – это позволяет сделать стандартная функция feof (проверка, достигнут ли конец файла; вернёт не 0, если прочитано всё).
5. Распространённый в примерах из Сети код вида
в ряде компиляторов может породить неточности при интерпретации данных. Например, этот код может прочитать как последнее значение завершающий перевод строки в файле, благодаря чему последнее прочитанное значение «удвоится».
В качестве примера форматной записи в файл сохраним массив a из 10 целочисленных значений в файле с именем result.txt по 5 элементов в строке:
Как и в случае с функциями для чтения форматированных данных, у всех этих методов имеются аналоги для работы со стандартным вводом/выводом.
Пример. Читая файл, определить длину каждой строки в символах. Для решения задачи воспользуемся тем фактом, что строки завершаются символом «перевод строки» ( ‘\n’ ). Предполагается, что файл уже открыт для чтения.
Пример. Читаем построчно файл с известной максимальной длиной строки. Предполагается, что файл уже открыт для чтения.
Пример. Целочисленный массив a запишем в двоичный файл.
Учитывая, что данные массива хранятся в последовательно идущих адресах памяти, цикл for для записи мы могли заменить одним оператором:
Подход к чтению данных с помощью fread аналогичен. Например, если файл уже открыт для чтения в режиме «rb»:
Пример. Определить размер файла в байтах, предположим, что файл уже открыт в режиме чтения или произвольного доступа.
Материал для чтения из пособия: пп. 8.6-8.11. Обратите внимание на таблицы с описанными прототипами функций ввода/вывода.
Рекомендуемые задачи: базовое задание включает две задачи, первая из которых предполагает обработку файла как текстовых данных, вторая – как бинарных. В качестве дополнительной третьей задачи может быть предложена реализация одной из задач 1, 2, содержащая консольный интерфейс и меню.
Про conio.h и почему его не надо использовать:
Stdio h что за библиотека
В приведенном ниже примере программа копирует символы, вводимые со стандартного ввода (клавиатуры), в указанный файл (файл задается первым параметром командной строки программы):
Использование fflush наиболее актуально при работе с текстовыми файлами, т.к. механизм буферизации откладывает запись в такой файл до получения символа конца строки (или до достижения конца буфера). Если приведенный выше пример модифицировать для копирования одного файла в другой, причем снабдить его прогрессом копирования (например, через каждые 1000 байт выводить символ *), то использование fflush становится обязательным, т.к. в противном случае звездочки прогресса будут выведены не по ходу копирования, а когда программа завершится или будет заполнен буфер.
Существует также набор простых процедур для ввода и вывода строк (под которыми понимается последовательность символов, завершаемая символом перевода строки). Эти процедуры удобно использовать в интерактивных программах, которые выполняют чтение с клавиатуры и вывод на экран терминала.
Приведенный ниже фрагмент программы содержит описание функции, возвращающей количество строк в указанном файле, при условии, что строки файла содержат не более 255 символов.
Примеры вызова функции printf :
Поле WIDTH определяет минимальное количество выводимых символов. При необходимости результат преобразования будет дополнен пробелами.
Поля WIDTH и PREC вместо числа могут содержать ‘*’, тогда в аргументах функции в соответствующем месте должен присутствовать целочисленный аргумент, значение которого и будет использоваться как параметр преобразования.
Примеры спецификаций:
Спецификация | Аргументы | Результат |
---|---|---|
%04X | i:integer=10 | |
j:integer=1000 | ||
%.3f | x:real=3.141592 | |
y:real=0.5 | ||
%+8.3G | x:real=3.141592 | |
y:real=-0.00005678 | ||
%.*Le | n:integer=2; x:extended=0.5 | |
m:integer=3; y:extended=1 | ||
%-*c%s | n:integer=3; c:char=’A’; s:pchar=’qwerty’ | |
m:integer=2; a:integer=64; s:pchar=’mail.ru’ |
Еще один комплексный пример:
Например, следующий оператор считывает очередное целое число из стандартного ввода:
Важно, что функции scanf передается адрес переменной i. Это необходимо для совместимости с языком Си, в котором параметры могут передаваться только по значению, а чтобы функция scanf изменяла переменную, которая находится в вызывающей процедуре, следует передать указатель, содержащий адрес этой переменной. Можно очень легко забыть про символ @, что приведет к ошибке записи в память. Нельзя также передавать неинициализированный указатель. Новичкам также приходится бороться с искушением помещать знак @ перед всеми указателями, такими как имена символьных массивов (pchar).
Поле WIDTH определяет максимальное количество символов входного потока, которое может быть использовано для преобразования по данной спецификации.
Если спецификация содержит знак ‘*’, это означает, что данной спецификации не требуется никакой аргумент, а результат преобразования нигде не будет сохранен.
Спецификация %[шаблон] определяет набор символов. До тех пор, пока очередной символ входного потока есть среди перечисленных в этом наборе, преобразование входного потока происходит по данной спецификации. Набор может задаваться простым перечислением символов: либо с использованием диапазонов: либо с использованием отрицания: Последний шаблон соответствует всем символам, кроме ‘a’, ‘b’ и ‘c’. Если требуется включить в шаблон символ ‘-‘, он должен стоять первым или последним символом шаблона. Если требуется включить в шаблон символ ‘^’, он должен стоять последним символом шаблона.
Примеры спецификаций:
Спецификация | Входной поток | Аргументы и результат |
---|---|---|
%d %2d %f %s | i:integer=11; j:integer=12; x:single=34.07; s1:array [0..N] of char=(‘k’,’e’,’i’,’t’,’h’,#0) | |
i:integer=123; j:integer=45; x:single=6; s1:array [0..N] of char=(‘4′,’.’,’7′,#0) | ||
%*d%1s%1c | s1:array [0..N] of char=(‘M’,’y’,#0); n:integer=2; s2:array [0..N] of char=(‘1′,’s’,’t’,’C’,’o’,’m’,’p’,#0) | s1:array [0..N] of char=(‘m’,’y’,’c’,’o’,’m’,’p’,#0); n:integer=6; s2:array [0..N] of char=(‘t’,’h’,’e’,’1′,’s’,’t’,#0) |
Еще один комплексный пример:
Stdio.h
stdio.h
stdio.h (от англ. standard input/output header — стандартный заголовочный файл ввода/вывода) заголовочный файл стандартной библиотеки языка Си, содержащий определения макросов, константы и объявления функций и типов, используемых для различных операций стандартного ввода и вывода. Функциональность унаследована от «портативного пакета ввода/вывода» («portable I/O package»), написанного Майком Леском из Bell Labs в начале 1970-х. [1] C++ ради совместимости, также использует stdio.h наряду со схожим по функциональности заголовочным файлом cstdio.
Содержание
Пример использования
Все функции в языке Си (и его многочисленных разновидностей) объявляются в заголовочных файлах. Таким образом, программистам приходится подключать файл stdio.h к исходному коду, чтобы использовать функции, объявленные в нем.
Программа, приведенная выше, считывает всю входящую информацию из стандартного ввода и выводит ее побайтно на стандартный вывод, добавляя символ перевода строки в конец вывода.
Функции-члены
Функции, объявленные в stdio.h в общем случае могут быть разделены на две категории: функции для операций с файлами и функции для операций ввода-вывода.
Имя | Примечания |
---|---|
Функции для файловых операций | |
fclose | закрывает файл, ассоциированный с переданным ей значением FILE * |
fopen, freopen, fdopen | открывают файл для определенных типов чтения и записи |
remove | удаляет файл (стирая его) |
rename | переименовывает файл |
rewind | работает аналогично fseek(stream, 0L, SEEK_SET), вызванному для потока, со сбросом индикатора ошибок |
tmpfile | создает и открывает временный файл, удаляемый при закрытии через fclose() |
Функции для операций ввода-вывода | |
clearerr | очищает EOF и индикаторы ошибок для данного потока |
feof | проверяет, установлен ли индикатор EOF для данного потока |
ferror | проверяет, установлен ли индикатор ошибок для данного потока |
fflush | принудительно записывает вывод, предназначенный для помещения в буфер, в файл, ассоциированный с данным потоком |
fgetpos | сохраняет позицию указателя файла потока, ассоциированный с его первым аргументом (FILE *), в его второй аргумент (fpos_t *) |
fgetc | возвращает один символ из файла |
fgets | получает строку из файла (оканчивающуюся символом перевода строки или конца файла) |
fputc | записывает один символ в файл |
fputs | записывает строку в файл |
ftell | возвращает указатель позиции файла, который может быть передан fseek |
fseek | производит смещение от текущей позиции в файле на указанное количество байт,или от его начала или конца, в указаном направление. |
fsetpos | устанавливает указатель позиции файла потока, ассоциированный с его первым аргументом (FILE *), как хранимый во втором его аргументе (fpos_t *) |
fread | читает данные из файла |
fwrite | записывает данные в файл |
getc | считывает и возвращает символ из данного потока и изменяет указатель позиции файла; позволяет использоваться как макрос с теми же эффектами, что и fgetc, кроме того, что может вычислять поток более одного раза |
getchar | имеет аналогичный эффект, что и getc(stdin) |
gets | считывает символы из stdin до символа перевода строки и хранит их в своем единственном аргументе |
printf, vprintf | используются для вывода в стандартный поток вывода |
fprintf, vfprintf | используются для вывода в файл |
sprintf, snprintf, vsprintf | используются для вывода в массив типа char (Строка в языке Си) |
perror | записывает сообщение об ошибке в stderr |
putc | записывает и возвращает символ в поток и изменяет указатель позиции файла на него; можно использовать как макрос с теми же свойствами, что и fputc, кроме того, что он может обрабатывать поток более одного раза |
putchar, fputchar | аналогичны putc(stdout) |
scanf, vscanf | используются для ввода из стандартного потока ввода |
fscanf, vfscanf | используются для ввода из файла |
sscanf, vsscanf | используются для ввода из массива char (то есть Строка в языке Си) |
setbuf | |
setvbuf | устанавливает режим буферизации для данного потока |
tmpnam | создает временное имя файла |
ungetc | помещает символ обратно в поток |
puts | выводит символьную строку в stdout |
Константы-члены
Следующие константы определены в заголовочном файле stdio.h :
Переменные-члены
Следующие переменные опеределены в заголовочном файле stdio.h :
Типы членов
Типы данных, определенные в заголовочном файле stdio.h содержат:
Stdio h что за библиотека
Это руководство поможет Вам разобраться в следующих вопросах использования printf с микроконтроллерами AVR:
1. Как перенаправить вывод printf на экран LCD, в UART, или любое другое пользовательское устройство вывода.
2. Как сконфигурировать стандартную библиотеку ввода/вывода (stdio), чтобы обрабатывались нужные опции форматирования printf (%X, %f и т. п.), или чтобы код обработки printf не занимал лишней памяти.
[Памятка для тех, кто все знает и кому лень читать дальше]
Пока это пустая болванка, которая ничего не умеет. По замыслу нужно, чтобы текст, который генерирует printf, выводился через USART.
Опциии линкера для поддержки printf, sprintf:
Опциии линкера для поддержки scanf, sscanf:
Примечание: при использовании тулчейна avr-gcc опция линкера обычно передается в makefile через макроопределение наподобие LD_FLAGS, LDOPTIONS, PRINTF_LIB и т. п. Пример опции для обработки полного формата (поддержка плавающей точки) функций сканирования scanf, sscanf:
Нужно подготовить низкоуровневую пользовательскую функцию (uart_putchar), которая будет непосредственно работать с аппаратурой вывода (в этом примере порт UART), т. е. получать и выводить символ текста. И конечно нужно не забыть сделать предварительную инициализацию аппаратуры вывода (init_uart).
Библиотека avrlibc имеет встроенное средство для того, чтобы связать Вашу функцию вывода с выходными данными функции printf:
[Макросы]
[Типы]
[Функции]
[Введение]
Файл stdio.h, подключаемый директивой #include, декларирует стандартные возможности ввода/вывода, реализованные в библиотеке avr-libc. Из-за внутреннего устройства нижележащей аппаратуры реализован только ограниченный набор стандартных средств ввода/вывода, которые обычно бывают доступны на «больших» системах. Здесь нет действительной поддержки файлов, есть только ввод/вывод устройства. Из-за того, что операционная система на AVR отсутствует, приложению нужно предоставить некую дополнительную информацию о том, как получить доступ подсистемам ввода/вывода микроконтроллера, чтобы их можно было использовать для стандартного I/O (см. далее макросы для организации стандартных потоков).
Также в среде выполнения AVR есть определенные ограничения по ресурсам (часто недостаточное количество памяти программ FLASH, оперативной памяти SRAM), поэтому некоторый функционал не реализован полностью (наподобие сложных преобразований printf). Тем не менее потенциальным пользователям подсистемы следует знать, что не смотря на ограничения реализации, функции семейства printf и scanf даже связанные с простейшим выводом типа «Hello, world!» в программе, в действительности довольно сложны, и займут довольно много места в памяти кода. Также они небыстрые из-за интерпретации (декодирования) строки формата во время выполнения. Т. е. к примеру, нечего и думать, чтобы использовать printf или scanf в микроконтроллерах типа ATtiny45 или ATmega8. Поэтому всякий раз, где это возможно, используйте другие, более простые, пусть даже не слишком универсальные (и возможно не стандартные, а написанные самостоятельно) подпрограммы преобразования в текст, которые будут потреблять меньше ресурсов (с точки зрения памяти и быстродействия).
Настраиваемые опции для кода против набора возможностей. Чтобы позволить программисту выбрать какой-то компромисс между функциональностью и затратами места под код, функция vfprintf() (которая является сердцем семейства функций printf) позволяет выбрать между разными вариантами обработки формата, используя опции линкера. Подробное описание см. в документации на функцию vfprintf(). Те же опции прикладываются и к vfscanf(), относящейся к семейству функций scanf.
Общая схема выбранного API. Предоставлены стандартные потоки stdin, stdout и stderr, но ограниченные стандартом C. Поскольку avr-libc ничего не знает о том, на каком устройстве работает, эти потоки не инициализированы при старте приложения. Кроме того, поскольку нет никакого понятия «file» применительно к avr-libc, то нет никакой функции fopen(), которую можно использовать для привязки к потоку (stream) на некотором устройстве (микроконтроллере), см. примечание 1. Вместо этого предоставлена функция fdevopen(), позволяющая привязать stream к устройству, где для устройства нужно предоставить функцию для отправки символа, приема символа, или и то, и другое. Библиотека avr-libc внутренне не делает никаких различий между текстовыми потоками («text» stream) и двоичными («binary» stream). Символ \n отправляется вниз к функции устройства put(). Если устройству нужно отправить символ перевода каретки (carriage return, \r) перед переводом строки (linefeed, \n), то подпрограмма устройства put() должна сама реализовать это (см. также примечание 2).
Для настройки предоставленной пользователем структуры FILE можно использовать для fdevopen() альтернативный метод, макрос fdev_setup_stream().
Для удобства первый вызов fdevopen(), который откроет поток (stream) для чтения, в результате выдаст поток, на который можно ссылаться по алиасу stdin. Аналогично первый вызов fdevopen(), который откроет поток на запись, свяжет его с обоими алиасами stdout и stderr. Также если открытие было сделано и на чтение, и на запись, то будут созданы три стандартных идентичных потока. Имейте в виду, что эти алиасы неотделимы друг от друга, так что если вызвать fclose() на таком потоке, то это также приведет к закрытию всех этих алиасов (см. примечание 3).
Можно привязать дополнительные данные пользователя к потоку, используя fdev_set_udata(). Backend-функции put и get могут после этого достать данные пользователя, используя fdev_get_udata(), и действовать соответствующим образом. Например, по этому принципу одна функция put может использоваться, чтобы осуществлять обмен с двумя разными UART-ами, или функции put и get могут здесь хранить внутреннее состояние между вызовами.
Строки формата в памяти программ (flash ROM). Все функции семейства printf и scanf поставляются в 2 вариантах, созданных для удобства: в виде стандартных имен, при этом подразумевается, что строки формата хранятся в SRAM, и также в виде версии с префиксом «_P», где ожидается, что строки формата размещены в flash ROM. Макрос PSTR (про который можно почитать в документации на [2]) очень удобен для декларирования этих строк формата.
Запуск stdio без malloc(). По умолчанию fdevopen() требует наличия malloc(). Как это часто нежелательно в условиях использования ограниченных по ресурсам микроконтроллеров, может быть предоставлена опция для запуска без использования malloc().
Предоставляется макрос fdev_setup_stream(), чтобы подготовить предоставленный пользователем буфер FILE для работы с stdio. Пример:
Этот пример использует форму инициализатора FDEV_SETUP_STREAM() вместо функции наподобие fdev_setup_stream(), так что все инициализации данных выполняется кодом start-up подсистемы языка C.
Если потоки, инициализированные таким способом, больше не нужны, то они могут быть уничтожены первым вызовом макроса fdev_close(), что уничтожит сам объект. Для таких потоков не должен использоваться вызов fclose(). В то время как вызов fclose() сам по себе безопасен, он приведет к неопределенной ссылке на free(), что приведет к привязке линкером модуля malloc к приложению.
Можно реализовать абстракцию устройства, которая совместима с fopen(), однако поскольку требуется сделать парсинг строки и нужно взять всю необходимую информацию либо из этой строки, либо из дополнительной таблицы, которая должна быть предоставлена приложением, то этот подход не был предоставлен.
Это обычно следует принципу организации Unix: если устройство, такое как терминал, нуждается в специальной обработке, то это область функционала, которую должен предоставить драйвер устройства. Таким образом, простая функция, которая подойдет в качестве put() для fdevopen(), и которая работает через интерфейс UART, может выглядеть примерно так:
[Документация на макросы]
Код возврата для конца файла (end-of-file) во время чтения устройства. Для использования в функции get после fdevopen().
Код возврата для события ошибки во время чтения устройства. Для использования в функции get после fdevopen().
fdev_setup_stream() на чтение
fdev_setup_stream() на чтение и запись
fdev_setup_stream() на запись
EOF декларирует значение, которое будет возвращено различными стандартными функциями ввода/вывода IO в случае ошибки. Поскольку платформа AVR (в настоящий момент) не содержит абстракции для действительных файлов, использование по назначению «end of file» (конец файла) здесь несколько бессмысленно.
Этот макрос освобождает любые ресурсы библиотеки, которые могут быть связаны с потоком. Он должен быть вызван, если поток больше не нужен, сразу перед тем, как приложение уничтожит сам объект. В настоящее время этот макрос вычисляется как пустота, но это может быть изменено в будущих версиях библиотеки.
Этот макрос получает указатель на данные, определенные пользователем, для объекта потока FILE.
Этот макрос вставляет указатель на данные, определяемые пользователем, в объект потока FILE. Данные пользователя могут быть полезны для отслеживания состояния в функциях put и get, предоставленных функцией fdevopen().
Настраивает предоставленный пользователем буфер как поток stdio. Этот макрос берет предоставленный пользователем буфер потока, и настраивает его как поток, который подходит для операций stdio, подобно тому, как если бы поток был получен динамически из fdevopen(). Буфер для настройки должен иметь тип FILE.
Аргументы put и get идентичны тем, что нужно передать функции fdevopen().
Аргумент rwflag может получить оно из значений _FDEV_SETUP_READ, _FDEV_SETUP_WRITE или _FDEV_SETUP_RW соответственно для чтения, записи или чтения/записи содержимого потока.
Примечание: функцией fdev_setup_stream() не будет выполнено назначение для стандартных потоков. Если используются стандартные потоки, то они нуждаются в назначении пользователем. См. также раздел «Запуск stdio без malloc()».
Инициализатор для потока stdio, предоставленного пользователем. Этот макрос работает подобно fdev_setup_stream(), однако он используется для инициализации переменной типа FILE.
Остальные используемые аргументы разъяснены в описании fdev_setup_stream().
Этот макрос getc используется как «быстрая» реализация макроса с функционалом, идентичным fgetc(). По соображениям экономии памяти в avr-libc он просто дает алиас на функцию fgetc.
Этот макрос getchar читает символ из stdin. Возвращаемые значения и обработка ошибок идентична fgetc().
Этот макрос putc используется как «быстрая» реализация для функционала, идентичного fputc(). По соображениям экономии памяти в avr-libc он просто дает алиас на функцию fputc.
Этот макрос putchar посылает символ c в stdout.
Поток, предназначенный для вывода ошибок. Если это не переназначено специально, идентичен stdout. Если stderr должен указывать на другой поток, результат другого fdevopen() должен быть явно назначен на этот макрос без закрытия предыдущего stderr (поскольку это также закроет stdout).
Поток, который будет использоваться как поток ввода для упрощенных функций, которые не получают поток в аргументах. Первый поток, открытый на чтение с использованием fdevopen() будет назначен на stdin.
Поток, который будет использоваться как поток вывода для упрощенных функций, не получающих поток в аргументах. Первый поток, открытый на запись с использованием fdevopen() будет привязан к обоим потокам stdin и stderr.
[Документация по функциям]
Очистка флагов ошибки и end-of-file потока.
Эта функция закрывает поток, что запрещает его дальнейшее использование для ввода/вывода. Когда Вы используете fdevopen() для настройки потока, нужно вызвать fclose() для завершения потока, чтобы освободить внутренние ресурсы.
Если поток был создан с использованием fdev_setup_stream() или FDEV_SETUP_STREAM(), вместо этого используйте fdev_close().
В настоящий момент fclose всегда возвращает 0 (успех).
Эта функция заменяет fopen(). Она открывает поток для устройства, где действительная реализация взаимодействия с устройством должна быть предоставлена приложением. Если результат успешный, то функция вернет указатель на структуру на открытый поток (stream). Причины для возможных возвратов ошибки в настоящий момент могут быть, если либо не предоставлены аргументы put или get, что означает открытие потока без возможности ввода/вывода как такового, или из-за недостатка динамической памяти, чтобы можно было создать новый поток.
Если предоставлен указатель на функцию put, поток открывается на запись. Функция, которая передана как аргумент put, должна принимать 2 аргумента: первый символ для записи в устройство, второй указатель на FILE, и должна вернуть 0 если вывод был успешным, и не ноль, если символ не был отправлен в устройство.
Если предоставлен указатель на функцию get, поток открывается на чтение. Функция, переданная как get, должна получать в единственном аргументе указатель на FILE, и возвращать символ из устройства, переданный как тип int. Если произошла ошибка при попытке чтения устройства, функция get должна вернуть _FDEV_ERR. Если произошло событие end-of-file во время чтения устройства, то get должна вернуть _FDEV_EOF.
Если предоставлены обе функции put и get, поток откроется на чтение и запись своего содержимого.
Первый открытый поток на чтение будет назначен на stdin, и первый открытый на запись будет назначен на оба потока stdout и stderr.
fdevopen() использует вызов calloc() (и следовательно malloc()), чтобы выделить ресурсы для хранения нового потока.
Проверка флага end-of-file потока. Этот флаг можно очистить только вызовом clearerr().
Проверка флага ошибки потока. Этот флаг можно очистить только вызовом clearerr().
Сброс буферов потока (flush stream). В действительности это пустая операция, предоставленная только для совместимости кода, так как стандартная реализация ввода/вывода в настоящий момент не выполняет какую-либо буферизацию.
Функция fgetc читает символ из потока. Она вернет символ, или EOF в случае наступления события end-of-file или события ошибки. Чтобы различать эти ситуации, должны использоваться функции feof() или ferror().
Читает некий объем данных из потока (как минимум 1 байт), пока не встретится символ новой строки, и сохранит принятые символы в буфере, на который указывает str. За исключением ситуации возникновении ошибки, строка будет завершена символом нуля (NUL character).
Если произошла ошибка, то функция вернет указатель NULL и установит флаг ошибки потока, который можно проверить, используя вызов ferror(). Иначе будет возвращен указатель на строку.
Функция fprintf выполняет форматированный вывод в поток. Подробнее см. vfprintf().
То же самое, что и fprintf(), но здесь используется хранение строки формата __fmt в памяти программ (FLASH).
Функция fputc отправляет символ c (хотя он предоставлен типом int) в поток. Функция fputc вернет символ, или EOF в случае возникновения ошибки.
Записывает в поток __stream строку, на которую указывает __str. При успехе вернет 0, или EOF в случае ошибки.
То же самое, что и fputs(), но здесь используется хранение строки __str памяти программ (FLASH).
Читает __nmemb объектов, каждый размером __size, из потока __stream в буфер, на который указывает __ptr. Вернет количество успешно прочитанных объектов, т. е. __nmemb за исключением случаев возникновения ошибки или конца данных на потоке (end-of-file). Для разделения этих случаев нужно использовать функции feof() и ferror().
Функция fscanf выполняет форматированный ввод, беря входные данные из потока. Подробнее см. vfscanf().
То же самое, что и fscanf(), но строка формата берется из памяти программ (FLASH).
Запишет __nmemb объектов в поток, каждый объект размером __size. На первый байт первого объекта указывает __ptr. Возвращает количество успешно записанных объектов, например __nmemb если не было ошибок при выводе.
То же самое, что и fgets(), за исключением того, что тут поток явно не указан, и подразумевается работа с потоком stdin, и завершающие символы новой строки (newline, если они есть) не будут сохранены в строке. Вызывающий код должен отвечать за достаточность места для хранения прочитанных символов.
Функция printf выполняет форматированный вывод в поток stdout. Подробнее см. vfprintf().
Вариант printf(), который строку формата __fmt берет из памяти программ.
Запись строки, на которую указывает str, и завершающий символ новой строки (newline) в поток stdout.
То же самое, что и puts(), но здесь строка находится в памяти программ.
Функция scanf выполняет форматированный ввод из потока stdin. Подробнее см. vfscanf().
Вариант scanf(), где строка формата __fmt находится в памяти программ.
Работает аналогично sprintf(), но вместо этого подразумевается, что s бесконечного размера, так что будет в __s преобразовано не более __n символов (включая завершающий символ NUL). Возвращает количество символов, которое было записано в __s, если в ней было достаточно места.
Вариант snprintf(), где строка формата __fmt находится в памяти программ.
Вариант printf(), который отправляет отформатированные символы в строку __s.
Вариант sprintf(), который для хранения строки форматирования __fmt использует память программ.
Функция sscanf выполняет форматированный ввод, читая входные данные из буфера, на который указывает buf. Подробнее см. vfscanf().
Вариант sscanf(), использующий строку формата __fmt из памяти программ.
Функция ungetc() проталкивает символ __c (преобразованный в unsigned char) обратно в поток ввода __stream. Этот вставленный обратно символ будет возвращен последующим чтением потока. В настоящий момент только один символ может быть вставлен в поток обратно. Функция ungetc() вернет символ, который был преобразован и вставлен, или EOF, если операция завершилась неудачно. Если значение аргумента __c равно EOF, то операция завершится по ошибке и поток останется нетронутым.
В функции vfprintf сосредоточен центральный функционал семейства функций printf. Она выводит значения в поток под управлением строки формата, переданной через аргумент __fmt. Действительные значения для печати передаются через список аргументов изменяемого размера.
Функция vfprintf вернет количество символов, которое было записано в поток, или EOF, если произошла ошибка. В настоящий момент такое может произойти только тогда, когда поток не был открыт на запись.
Строка формата состоит их 0 или большего количества директив: обычные символы (не %), которые будут без изменения копироваться в выходной поток; и спецификации преобразования, каждая из которых выдает преобразование от 0 или большего количества последующих аргументов списка. Каждая спецификация преобразования вводится символом %. Аргументы должны правильно соответствовать (по своему типу) спецификатору преобразования. После % появляется следующая последовательность символов:
• 0 или большее количество следующих флагов:
# значение, которое должно быть преобразовано к «альтернативной форме». Для преобразований c, d, i, s, и u эта опция не дает никакого эффекта. Для преобразований o точность числа увеличивается так, чтобы количество выводимых символов увеличивалось (за исключением печати значения 0, которое имеет явную точность 0). Для преобразований x и X, не нулевой результат дает строку ‘0x’ (или ‘0X’ для преобразований X).
0 (zero) дополнение слева нулями. Для всех преобразований, преобразованное значение будет дополнено слева нулями, а не пробелами. Если указана точность для числовых преобразований (d, i, o, u, i, x и X), флаг 0 игнорируется.
— флаг отрицательного поля ширины; преобразуемое значение получает левое выравнивание по границе поля. Преобразованное значение дополняется справа пробелами, а не слева пробелами или нулями. Этот флаг отменяет 0, если они предоставлены оба.
‘ ‘ (пробел) перед положительным числом должен быть пробел, когда преобразуется число со знаком(d или i).
+ знак должен быть всегда размещен перед числом, когда выполняется преобразование числа со знаком. Этот флаг отменяет флаг ‘пробел’, если они указаны вместе.
• Необязательная строка цифр, указывающая минимальную длину поля. Если преобразованное значение короче, чем указанная ширина поля, то она будет дополнена слева пробелами (или справа, если дан флаг левого выравнивания), чтобы заполнить поле до указанной ширины.
• Необязательная точность (precision), указанная через точку ‘.’, за которой идет необязательная строка цифр. Если строка цифро опущена, то подразумевается точность 0. Это дает минимальное количество цифр для преобразований d, i, o, u, x и X, или максимальное количество символов для вывода преобразований s.
• Необязательный модификатор длины l или h, который указывает, что аргумент для преобразований d, i, o, u, x или X означает «long int» вместо int. Модификатор h игнорируется, поскольку «short int» эквивалентен int.
• Символ, который указывает тип преобразования (conversion specifier).
Conversion specifier может быть следующий:
• d, i, o, u, x или X. Целый аргумент (int или подходящий вариант) преобразуется как знаковое десятичное (signed decimal, для d и i), беззнаковое восьмеричное (unsigned octal, для o), беззнаковое десятичное (unsigned decimal, для u), или беззнаковое шестнадцатеричное (unsigned hexadecimal, для x и X). Символы «abcdef» (маленькие) используются для преобразований x; буквы «ABCDEF» (большие) используются для преобразований X. Точность (precision) если она указана, дает минимальное количество появляющихся цифр; если у преобразованного значения получается меньше цифр, то слева добавляются нули.
• p. Аргумент void * берется как беззнаковое целое (unsigned), и преобразуется так же, как по формату %#x.
• c. Аргумент int преобразуется в argument unsigned char, и будет записан символ с результирующим кодом.
• s. Аргумент char * интерпретируется как указатель на массив символов (указатель на строку). Символы из массива записываются до завершающего NUL (но сам NUL отбрасывается); если указана точность (precision), то будет записано не больше символов, чем указано. Если дана точность, то наличия в строке завершающего нуля не обязательно; если точность не указана, или если она больше, чем размер массива, то массив должен содержать в себе символ завершения строки (0, символ NUL).
• %. Будет записан %. Нет аргументов для преобразования. Используется, если нужно вывести символ процента. Полная спецификация формата «%%».
• e, E. Аргумент double будет округлен и преобразован в формат «[-]d.dddeA±dd», где одна цифра до десятичной точки, и количество цифр после десятичной точки равно точности (precision); если точность не указана, то берется точность 6; если указана точность 0, то не будет использоваться десятичная точка. Преобразование E использует букву ‘E’ (большая, а не маленькая ‘e’) для указания экспоненты exponent. Экспонента всегда содержит 2 цифры; если значение равно 0, то экспонента 00.
• f, F. Аргумент double округляется и преобразуется в формат «[-]ddd.ddd», где количество цифр после десятичной токи равно указанной точности (precision). Если точность не указана, то берется точность 6; если точность явно указана 0, то не будет десятичной точки и цифр после неё. Если появляется десятичная точка, то перед ней будет указана хотя бы одна цифра.
• S. То же самое, что и формат s, за исключением того, что ожидается указатель на место в памяти программ (ROM, FLASH), а не в RAM.
Ни в коем случае не будет сделано усечение по причине указания слишком малого поля ширины; если результат преобразования больше, чем поле ширины, то поле будет расширено до того, чтобы вместить результат преобразования.
[Управление библиотеками stdio через опции линкера]
Поскольку полная реализация всех вышеупомянутых возможностей становится довольно велика по объему, то имеется 3 разные варианта реализации vfprintf(), которые можно выбрать опциями линкера. По умолчанию vfprintf() реализует все упомянутые возможности, за исключением преобразований для чисел с плавающей точкой. Минимизированная версия vfprintf() реализует только базовую обработку целых чисел и строк, но с флагами преобразования можно указать только дополнительную опцию # (эти флаги корректно обрабатываются по спецификации формата, но просто игнорируются). Минимальная версия обработки может быть запрошена следующими опциями компилятора:
Если нужна полная функциональность, включая обработку преобразований floating point, то нужно использовать следующие опции:
Есть ограничения: указанная ширина и точность должна быть самое большее 255.
• Модификатор длины hh игнорируется (аргумент char интерпретируется как int). Точнее, эта реализация не проверяет количество символов h в формате.
• Однако модификатор длины ll оборвет вывод, так как эта реализация не обрабатывает аргументы long long.
• Поле переменной ширины или точности (variable width, precision field с символом *) в этой версии не реализовано, и такой формат оборвет вывод.
Вариант vfprintf(), который использует строку формата из памяти программ.
[Форматированный ввод]
Эта функция является основой семейства функций scanf. Символы читаются из потока и обрабатываются так, как это описано в fmt. Результат преобразований будет присвоен параметрам, переданным через ap.
Строка формата fmt сканируется в поиске спецификаций преобразования. Все, что не соответствует спецификации формата, берется как текст, буквально поступающий со входа. Пробелы в строке формата будут соответствовать пробелам в данных (пробелов может и не быть), все другие символы соответствуют только сами себе. Обработка будет оборвана, как только данные или строка формата больше не соответствуют, или произошла ошибка или событие конца данных на потоке (события error или end-of-file).
Большинство преобразований пропускают лидирующие пробелы перед началом действительного преобразования. Спецификаторы преобразования вводятся символом %. За знаком % могут идти следующие опции:
• * означает, что преобразование должно быть выполнено, но результат должен быть отброшен; из ap не будут обработаны параметры.
• h означает, что аргумент указывает на short int (а не на int).
• hh означает, что аргумент указывает на char (а не на int).
• l показывает, что аргумент указывает на long int (а не на int для преобразований целого типа) или на double (для преобразований floating point).
Дополнительно может быть указано поле максимальной ширины как ненулевое положительное целое число, которое ограничит преобразование на указанное самое большое количество символов из входного потока. Это поле ширины ограничено 255 символами, что является также значением по умолчанию (за исключением преобразования c, у которого умолчание ширины равно 1).
Поддерживаются следующие флаги преобразования:
• %. Соответствует просто символу %. Это не преобразование.
• d. Соответствует опционально знаковому десятичному целому числу; следующий указатель в аргументах должен указывать на int.
• i. Соответствует опционально знаковому десятичному целому числу; следующий указатель в аргументах должен указывать на int. Целое число читается по базе 16, если оно начинается на 0x или 0X, по базе 8 если начитается на 0, иначе по базе 10. Используются только те символы, которые соответствуют базе представления чисел.
• o. Соответствует целому восьмеричному; указатель в аргументах должен быть указателем на unsigned int.
• u. Соответствует на опционально знаковое десятичное целое; указатель в аргументах должен указывать на unsigned int.
• x. Соответствует опционально знаковому шестнадцатеричному целому; указатель в аргументах должен указывать на unsigned int.
• f. Соответствует опционально знаковому числу с плавающей точкой; указатель в аргументах должен указывать на float.
• e, g, F, E, G. Эквивалент f.
• c. Соответствует последовательности с неким количеством символов (по умолчанию 1 символ); указатель в аргументах должен указывать на char, и должно быть достаточно места для всех символов (завершающий NUL добавлен не будет). Подавляется обычный пропуск лидирующих пробелов. Чтобы пропустить лидирующие пробелы, используйте в формате явно указанные пробелы.
• [. Соответствует не пустой последовательности символов от указанного набора допустимых символов; указатель в аргументах должен указывать на char, и должно быть достаточно места для всех символов в строке, плюс завершающий символ NUL. Подавляется обычный пропуск лидирующих пробелов. Последовательность будет составлена из символов этого набора (кроме тех, которых нет в наборе); набор определяется символами между открывающей скобкой [ и закрывающей скобкой ]. Набор исключает те символы, если первый символ после открывающей скобки был ^. Чтобы добавить в набор закрывающую скобку, сделайте её первой после открывающей скобки или ^; любая другая позиция для закрывающей скобки будет означать конец спецификации набора. Символ дефиса ‘-‘ также носит специальный смысл; когда он размещен между двумя символами, то это добавит в набор все символы, которые находятся по коду между ними. Чтобы добавить в набор сам дефис, сделайте его последним в наборе, указав перед закрывающей скобкой. Например, [^]0-9-] любой набор символов кроме закрывающей скобки, всех цифр от 0 до 9 и дефиса. Строка, которая завершится появлением любого символа не в наборе (или, если применен ^, то в наборе), или когда поле ширины закончится. Имейте в виду, что использование этого преобразование увеличивает использование памяти под стек.
• p. Соответствует значению указателя (как напечатано для p в printf()); указатель в аргументах должен указывать на void.
• n. Ничего не ожидается; вместо этого будет выбрано из ввода некоторое количество символов, и помещено по следующему указателю в аргументах, который должен быть указателем на int. Это не будет преобразованием, хотя оно может быть подавлено флагом *.
Эти функции возвратят количество назначенных входных элементов в аргументах, количество которых может быть меньше или даже 0, когда даст сбой совпадение. 0 говорит о том, что ни одна входная переменная не была назначена; обычно это происходит из-за несоответствующего входного символа, к примеру был алфавитный символ (буква) для преобразования d. Будет возвращено значение EOF, если на входе была ошибка до того, как произошло любое из преобразований, как например произошло событие окончания данных на потоке (end-of-file). Если произошло событие end-of-file уже после того, как преобразование началось, то будет возвращено действительное количество успешных преобразований.
По умолчанию допустимы все вышеописанные преобразования, за исключением преобразований floating-point и ширины, ограниченной 255 символами. Преобразование float-point будет доступно в расширенной версии, предоставленной библиотекой libscanf_flt.a. Также в этом случае ширина не ограничена (на самом деле есть ограничение в 65535 символов). Чтобы прилинковать к программе расширенную версию, используйте для стадии линковки следующие флаги компилятора:
Третья версия доступна для случаев работы в ограниченных по памяти условиях. В дополнение к ограничениям стандартной версии, эта версия не реализует спецификацию %[. Она предоставлена в библиотеке libscanf_min.a, и может быть запрошена следующими опциями на стадии линковки:
Вариант vfscanf(), использующий строку формата из памяти программ.
Функция vprintf выполняет форматированный вывод в поток stdout, получая тот же список переменных аргумента, как и в функции vfprintf(). Подробнее см. описание vfprintf().
Функция vscanf выполняет форматированный ввод из потока stdin, получая тот же список переменных в аргументах, как и vfscanf(). Подробнее см. описание vfscanf().
То же самое, что и vsprintf(), но вместо строки бесконечного размера здесь подразумевается, что s имеет конечный объем, так что в строку s не может быть преобразовано больше n символов (включая символ NUL). Функция возвратит количество записанных символов, если было достаточно свободного места.
Вариант vsnprintf(), который использует строку формата из памяти программ.
То же самое, что и sprintf() но получает в аргументах список переменных, который может быть произвольной длины.
Вариант vsprintf(), который использует строку формата из памяти программ.