04b-variadic (1238877)
Текст из файла
VARIADICМинимум о функциях с переменным числом аргументовК. Владимиров, Intel, 2019mail-to: konstantin.vladimirov@gmail.comПеревод в строку и конкатенация•Постановка задачи: нужно собрать строчку из строки, двух чисел и ещё однойстроки// "ab", 42, 1, "cd" → "ab 42 1 cd"void strange_concat(char *dest, char const *s1,int d1, int d2, char const *s2) {// TODO: ваш код здесь}•Будем считать, что размер dest заведомо достаточен•Как бы вы это сделали?Загадочный sprintf•Самый простой способ: просто напечатать всё это в строку// "ab", 42, 1, "cd" → "ab 42 1 cd"void strange_concat(char *dest, char const *s1,int d1, int d2, char const *s2) {sprintf(dst, "%s %d %d %s", s1, d1, d2, s2);}•Функция sprintf удивительно обобщённая: она позволяет скидывать всё чтоугодно в строчку и часто пользоваться ей удобнее, чем специфичными•Хорошо.
Но как написать саму функцию sprintf?•Даже проще: что самое удивительное в функции sprintf?Вариабельные функции•Самое удивительное в функции sprintf то, что она берёт сколько угодноаргументов•Давайте сначала попробуем написать функцию, которая брала бы сколькоугодно целых чисел и складывала ихint x = sum_all(4, 10, 14, 24, 40); // x == 88•Первый параметр это количество аргументов (иначе откуда его узнать?)•Кажется её логика попрощеВариабельные функции•Произвольное количество аргументов после троеточияint sum_all (int n, ...) {int res = 0;// здесь нужно просуммировать все аргументыreturn res;}•Здесь три точки это не сокращение на слайде, это легальный синтаксис•Остаётся вопрос как всё-таки поулчить доступ к аргументам?Функции из stdarg•Список аргументов создаётся через va_listva_list args;•Аргумент с которого начинаются вариабельные отмечается через va_startva_start(args, n);•Каждый аргумент вынимается через va_arg с указанием типаva_arg(args, int);•В конце всё завершается через va_endva_end(args);Пример: суммирование целых•Собираем всё вместе: функция суммирует целые числаint sum_all(int n, ...) {int res = 0;va_list args;va_start(args, n);for (int i = 0; i < n; ++i)res += va_arg(args, int);va_end(args);return res;}•Теперь заработает: x = sum_all(4, 10, 14, 24, 40); // x == 88Именно так работают printf и scanf•Функции printf и scanf объявлены следующим образомint printf(const char *format, ...);int scanf(const char *format, ...);•Они тоже принимают произвольное число параметров и используют строкуформата чтобы установить типы•Любая ошибка в типах ведёт к непоправимым последствиям•И конечно, именно так работает и sprintf•Но прежде чем мы до него дойдём, ещё одно простое применениеВариабельная функция-конструктор•Все помнят (см.
Problem MP) что полином мы представляем как:struct Poly { unsigned n; int *p; };•Можно написать вариабельную функцию-конструктор полинома• = 3 + 3 2 + 4 + 7struct Poly create_poly (unsigned n, ...) {// TODO: выделить память и заполнить её}struct Poly A = create_poly(4, 1, 3, 4, 7);•Напишите эту функцию!Многоликий printf и scanf•Основные формы:int printf(const char *format, ...);int scanf(const char *format, ...);int fprintf(FILE *f, const char *format, ...);int fscanf(FILE *f, const char *format, ...);int sprintf(char *s, const char *format, ...);int sscanf(char *s, const char *format, ...);•По сути обычный printf это fprintf где вместо первого аргумента stdoutОбсуждение•Можем ли мы имея fprintf написать printf?int printf(const char *format, ...) {// как-то вызвать fprintf}Обсуждение•Можем ли мы имея fprintf написать printf?int printf(const char *format, ...) {// как-то вызвать fprintf}•Увы, в языке нет способа из функции "пробросить троеточие"•Можно написать макрос, но мы хотим избежать макросов•А что если передать va_list?Волшебство vfprintf•Теперь и printf и fprintf можно реализовать в терминах новой функцииint vfprintf(FILE *f, const char *format, va_list arg);int fprintf(FILE *f, const char *format, ...) {va_list l; int retval;va_start(l, format);retval = vfprintf(f, format, l);va_end(l);return retval;}int printf(const char *format, ...) // как-то вызвать vfprintfОбсуждение•Функции, такие как vfprintf и vsprintf очень полезны при написаниисобственных printf-подобных функцийpFile = fopen (szFileName,"r");if (pFile == NULL)PrintFError("Error opening '%s'", szFileName);•Понятно, что здесь PrintFError должна как-то вызвать внутри perror, но как еёможно реализовать?Обсуждение•Функции, такие как vfprintf и vsprintf очень полезны при написаниисобственных printf-подобных функцийvoid PrintFError(const char * format, ...) {char buffer[256];va_list args;va_start(args, format);vsprintf(buffer,format, args);perror(buffer);va_end(args);}•Эта реализация не слишком совершенна (а что если буфер переполнится?), новполне обычна для языка C.
Характеристики
Тип файла PDF
PDF-формат наиболее широко используется для просмотра любого типа файлов на любом устройстве. В него можно сохранить документ, таблицы, презентацию, текст, чертежи, вычисления, графики и всё остальное, что можно показать на экране любого устройства. Именно его лучше всего использовать для печати.
Например, если Вам нужно распечатать чертёж из автокада, Вы сохраните чертёж на флешку, но будет ли автокад в пункте печати? А если будет, то нужная версия с нужными библиотеками? Именно для этого и нужен формат PDF - в нём точно будет показано верно вне зависимости от того, в какой программе создали PDF-файл и есть ли нужная программа для его просмотра.