sem07 (1114915), страница 2
Текст из файла (страница 2)
Указание имени поля или индекса массива позволяет записывать инициализаторы в произвольном порядке. Например:void func(int n){struct circle cc = { .r = 10, .x = 4 };int m[3] = { [1] = 1, [0] = 2 };}Неинициализированные элементы локальных объектов, как обычно, будут содержатьнеопределённые значения. Каждое поле или индекс массива может быть проинициализирован только раз.
Если встречаются и инициализаторы с указанием индекса или поля и безних, индекс или имя поля вычисляется автоматически.void func(int n){int m[3] = { [1] = 1, 2 };}В данном примере элемент m[0] будет не инициализирован, элемент m[1] получит значение 1, а m[2] — 2.Другое новшество C99 — возможность использования в выражениях составных литералов структурного и массивового типов.
Составной литерал записывается следующимобразом:(<тип>) <инициализатор>напримерcc = (struct circle) { x + 1, y + 1, 5 };Здесь выражения в инициализаторе также могут быть неконстантными.Использование составного литерала эквивалентно созданию в текущей области видимости анонимной переменной соответствующего типа и её начальной инициализации указанным инициализатором. Анонимная переменная создаётся только один раз.1.3 Передача параметров в программуПри запуске программы операционная система передаёт ей параметры, которые былиуказаны в командной строке. Если программа желает работать с переданными ей параметрами, заголовок функции main должен выглядеть следующим образом:int main(int argc, char *argv[]);Здесь первый параметр argc задаёт количество аргументов командной строки, а параметр argv — это массив указателей на строки, хранящие значения соответствующих аргументов.
По соглашению argv[0] содержит само имя программы. Следовательно, если программе не было передано никаких аргументов, argv равно 1. Разбиение командной строкина аргументы запускаемой программы производит командный процессор, обычно аргументыразделяются друг от друга пробелами. Например, если программа была запущена командой5./prog1 f1 f2 ../f3argc пример значение 4, argv[0] указывает на строку "./prog1", argv[1] — на строку"f1" и т. д.Значение, возвращаемое функцией main (или аргумент функции exit), — это «код возврата» программы.
Он используется чтобы проинформировать вызвавшую программу о статусе завершения работы программы. Код возврата 0 означает, что работа программы завершилась нормально, ненулевые коды возврата означают, что при выполнении программывозникли какие-то проблемы.Пример программы, которая печатает свои аргументы.123456789101112#include <stdio.h>intmain(int argc, char *argv[]){int i;for (i = 0; i < argc; i++) {printf("argv[%d] = %s\n", i, argv[i]);}return 0;}1.4 Функции работы с файлами и потокамиМы ещё не рассмотрели две функции для посимвольной работы со стандартным потоком.Функцияint getchar(void);вводит один символ из стандартного входного потока.
Если символ введён успешно, возвращается код символа в диапазоне 0–255 (положительное число). В случае ошибки иликонца файла возвращается значение EOF (оно обычно равно -1). Функцияint putchar(int c);выводит один символ на стандартный поток вывода.Обратите внимание, что функция getchar может возвращать 257 различных значений,поэтому переменной типа char для хранения возвращаемого значения недостаточно! Следующий пример копирует стандартный ввод на стандартный вывод.1234567#include <stdio.h>int main(){int c;while ((c = getchar()) != EOF) putchar(c);return 0;}Стандартная библиотека содержит средства для работы с произвольными файлами.
Дляхранения информации об открытом файле используется структура FILE. Функции работы сфайлами принимают или возвращают указатель на эту структуру. ФункцияFILE *fopen(char *name, char *mode);6открывает файл с именем name. Строка mode содержит флаги, с которыми открываетсяфайл."r"открыть для чтения. Текущая позиция в файле устанавливается на началофайла."w"открыть для записи. Если файл не существовал, он создаётся, если существовал, он очищается. Текущая позиция в файле устанавливается на началофайла."a"открыть для добавления.
Если файл не существовал, он создаётся. Текущаяпозиция в файле устанавливается на конец файла."r+" открыть для чтения и записи. Текущая позиция в файле устанавливается наначало файла."w+" открыть для чтения и записи. Если файл не существовал, он создаётся, еслисуществовал, он очищается. Текущая позиция в файле устанавливается наначало файла."a+" открыть для чтения и записи. Если файл не существовал, он создаётся. Текущая позиция в файле устанавливается на конец файла.Если файл был открыт успешно, возвращается указатель на структуру, хранящую состояние открытого файла.
Если при открытии произошла ошибка, возвращается NULL.int fclose(FILE *stream);Закрывает поток. Если закрытие прошло успешно, возвращается 0, иначе возвращаетсяEOF.При запуске программы открываются стандартные потоки stdin, stdout, stderr.Например, функции putchar, puts работают с потоком stdin, а функции getchar, getsработают с потоком stdout.
Вы можете использовать эти имена для указания имён стандартных потоков в функциях, перечисленных ниже.intintintintgetc(FILE *stream);putc(int c, FILE *stream);fscanf(FILE *stream, char *format, ...);fprintf(FILE *stream, char *format, ...);Соответствуют функциям getchar, putchar, scanf, printf.Функцияint fputs(char *str, FILE *stream);записывает строку символов str в файл. Символ-терминатор строки отбрасывается. Вотличие от puts эта функция не дописывает ’\n’ в выходной файл.Функцияchar *fgets(char *str, int size, FILE *stream);Считывает самое большее size - 1 символ из входного файла. Чтение останавливается по достижению конца файла или по достижению символа ’\n’.
Если ’\n’ считан, ондобавляется в строку. После прочитанных символов добавляется символ-терминатор ’\0’.При успешном завершении функция возвращает str, а при неудаче (конец файла или ошибка ввода) возвращается NULL. Используйте эту функцию для чтения строк из входного файла, поскольку эта функция ограничивает максимальную длину считываемойстроки.Следующий пример копирует содержимое заданного файла на стандартный вывод.712345678910111213141516171819#include <stdio.h>int main(int argc, char *argv[]){FILE *f;char buf[256];if (argc != 2) {fprintf(stderr, "Wrong number of arguments\n");return 1;}if (!(f = fopen(argv[1], "r"))) {fprintf(stderr, "Cannot open file %s\n", argv[1]);return 1;}while (fgets(buf, sizeof(buf), f))fputs(buf, stdout);fclose(f);return 0;}Функцияint ungetc(int c, FILE *stream);Заталкивает один символ обратно во входной поток, так что следующая функция getcсчитает именно этот символ.
Таким образом затолкнуть назад можно не более одного символа.Функцияint feof(FILE *stream);возвращает ненулевое значение, если в структуре FILE установлен флаг ошибки или конца файла. Этот флаг устанавливается по результату работы функций чтения или записи. До первого чтения флаг сброшен, даже если файл пуст. В следующем примере последняястрока будет напечатана дважды.12345678910#include <stdio.h>int main(int argc, char *argv[]){char buf[64];while (!feof(stdin)) { // неверно!!!fgets(buf, sizeof(buf), stdin);fputs(buf, stdout);}return 0;}1.5 Упражнения1.
В аргументах командной строки задаются целые числа. Необходимо просуммировать их и напечатать результат на стандартный поток вывода.2. В аргументах командной строки задаются имена текстовых файлов, содержащих последовательности вещественных файлов. Для каждого входного файла с именем Name создать выходной файл с именем Name.max и записать в него максимальное значение во входном файле.8.