
- •1.1 Перші кроки
- •1.2 Змінні й арифметичні вирази
- •1.3 Твердження for
- •1.4 Символічні константи
- •1.5 Ввід і вивід знаків
- •1.5.1 Копіювання файла
- •1.5.2 Відлік символів
- •1.5.3 Відлік рядків
- •1.6 Масиви
- •1.7 Функції
- •1.8 Аргументи - виклик за значенням
- •1.9 Символьні масиви
- •1.10 Зовнішні змінні й область дії
- •2.1 Назви змінних
- •2.2 Типи даних і розміри
- •2.3 Константи
- •2.4 Оголошення
- •2.5 Арифметичні операції
- •2.6 Реляційні та логічні оператори
- •2.7 Перетворення типів
- •2.8 Оператори приросту та спаду
- •2.9 Розрядні оператори
- •2.10 Оператори та вирази присвоєння
- •2.11 Вирази умов
- •2.12 Пріоритет і послідовність обчислення
- •3.1 Вирази та блоки
- •3.3 Else if
- •3.4 Switch
- •3.5 Цикли while та for
- •3.6 Цикли do-while
- •3.7 Break і continue
- •3.8 Goto та мітки
- •4.1 Основні знання про функції
- •4.2 Функції, які не повертають цілих
- •4.3 Зовнішні змінні
- •4.4 Правила області дії
- •4.5 Файли заголовка
- •4.6 Статичні змінні
- •4.7 Регістрові змінні
- •4.8 Структура блоків
- •4.10 Рекурсія
- •4.11 Препроцесор c
- •4.11.1 Включення файлів
- •4.11.2 Заміна макросів
- •4.11.3 Обумовлене включення файлів
- •5.1 Покажчики й адреси
- •5.2 Покажчики й аргументи функцій
- •5.3 Покажчики та масиви
- •5.4 Арифметика адрес
- •5.5 Покажчики на символи та функції
- •5.6 Масив покажчиків; покажчики на покажчики
- •5.7 Багатовимірні масиви
- •5.8 Ініціалізація масиву покажчиків
- •5.9 Покажчики в порівнянні з багатовимірними масивами
- •5.10 Аргументи командного рядка
- •5.11 Покажчики на функції
- •5.12 Складні оголошення
- •6.1 Основні поняття про структури
- •6.2 Структури та функції
- •6.3 Масиви структур
- •6.4 Покажчики на структури
- •6.5 Структури зі зворотнім звертанням
- •6.6 Пошук по таблиці
- •6.7 Typedef
- •6.8 Сполуки
- •6.9 Розрядні поля
- •7.1 Стандартний ввід і вивід
- •7.2 Форматований вивід - printf
- •7.3 Списки аргументів довільної довжини
- •7.4 Форматований ввід - scanf
- •7.5 Доступ до файлів
- •7.6 Обробка помилок - stderr і exit
- •7.7 Ввід і вивід рядків
- •7.8 Додаткові функції
- •7.8.1 Операції з ланцюжками
- •7.8.2 Перевірка і перетворення класів символів
- •7.8.3 Ungetc
- •7.8.4 Виконання команд
- •7.8.5 Керування пам'яттю
- •7.8.6 Математичні функції
- •7.8.7 Генератор випадкових чисел
- •8.1 Дескриптори файлів
- •8.2 Низькорівневий ввід/вивід - read і write
- •8.3 Open, creat, close, unlink
- •8.4 Довільний доступ - lseek
- •8.5 Приклад: втілення fopen і getc
- •8.6 Приклад - перелік вмісту каталогів
- •8.7 Приклад - розподільник пам'яті
7.3 Списки аргументів довільної довжини
Цей розділ містить втілення мінімальної версії printf для демонстрації того, як написати функцію, яка би обробляла список аргументів довільної довжини у машинонезалежний спосіб. Оскільки ми, головним чином, зацікавлені в опрацьовуванні аргументів, minprintfоброблятиме форматувальний ланцюжок і аргументи, але викликатиме дійсний printf для перетворень формату. Чинним оголошенням printf є
int printf(char *fmt, ...)
де ... означає, що кількість і типи цих аргументів може відрізнятися. Оголошення ... може тільки з'являтися в кінці списку аргументів. Нашу minprintf оголошено як
void minprintf(char *fmt, ...)
оскільки ми не повертатимемо відлік символів, як це робить printf.
Делікатний момент полягає в тому, як minprintf просувається через список аргументів, в той час як сам список навіть не має імені. Стандартний файл заголовка <stdarg.h> містить набір макросів, який визначає, як проходити список аргументів. Втілення цього файла заголовка може відрізнятися на різних машинах, але інтерфейс, який він представляє є однорідним.
Тип va_list використовується для оголошення змінної, яка посилатиметься по-черзі до кожного аргументу; в minprintf ця змінна називається ap, як скорочення від "argument pointer" (покажчик на аргумент). Макрос va_start ініціалізує ap таким чином, щоб вона вказувала на перший безіменний аргумент. Макрос потрібно викликати один раз до того, як використовувати ap. Необхідно, щоб був щонайменше один аргумент з іменем; останній названий аргумент використовується va_start, щоб розпочати роботу.
Кожний виклик va_arg повертає один аргумент і переводить ap до наступного; va_argвикористовує назву типу, щоб визначити, який тип повернути і наскільки великим повинен бути крок. І, накінець, va_end здійснює очистку, якщо треба. Її потрібно викликати до того, як програма поверне своє значення.
Ці властивості складають базу для нашої спреощеної printf:
#include <stdarg.h>
/* minprintf: мінімальна printf зі списком аргументів *
* змінної довжини */
void minprintf(char *fmt, ...)
{
va_list ap; /* покажчик на кожний безімений аргумент *
* по черзі */
char *p, *sval;
int ival;
double dval;
va_start(ap, fmt); /* заставляє ap бути покажчиком *
* на 1-ий безіменний аргумент */
for (p = fmt; *p; p++) {
if (*p != '%') {
putchar(*p);
continue;
}
switch (*++p) {
case 'd':
ival = va_arg(ap, int);
printf("%d", ival);
break;
case 'f':
dval = va_arg(ap, double);
printf("%f", dval);
break;
case 's':
for (sval = va_arg(ap, char *); *sval; sval++)
putchar(*sval);
break;
default:
putchar(*p);
break;
}
}
va_end(ap); /* очистка */
}
Вправа 7-3. Виправіть minprintf так, щоб вона втілювала додаткові можливості printf.