sem03 (1114911), страница 2
Текст из файла (страница 2)
После этого в buf будет добавленсимвол-терминатор ’\0’. Обратите внимание на отсутствие знака ’&’ перед buf в аргументах вызова scanf — массивы и так передаются по ссылке. Буфер buf должен бытьдостаточного размера, чтобы вместить все символы вводимой строки. Поскольку вводимыеданные чаще всего не находятся под контролем программиста, то есть пользователь, возможно злонамеренно, может вводить строки произвольной длины, использование такогонеконтролируемого чтения в реальных программах очень опасно.
Задать ограничениена размер считываемой строки можно следующим образом:char buf[20];scanf("%19s", buf);В данном случае в строку buf будет считано не более 19 символов, и в конец строки будетдобавлен байт-терминатор ’\0’. Поэтому никогда не используйте спецификатор %sбез ограничения размера вводимой строки!.Другой полезный спецификатор для scanf — %n. Этому спецификатору должна соответствовать переменная целого типа со знаком & (адрес). В эту переменную заносится числосимволов, прочитанное к моменту, когда был встречен этот спецификатор. Спецификатор%n не учитывается в числе успешно считанных спецификаторов в возвращаемом значениифункции scanf.Очень полезной является функция sprintf с прототипомint sprintf(char s[], char format, ...);Которая работает точно также, как printf, поддерживает все форматы вывода, но вместопечати на стандартный поток вывода, заполняет строку s.
В конец строки s добавляетсясимвол-терминатор. Результатом работы функции является количество выведенных символов (не считая символ-терминатор). Например,sprintf(s, "%d", n);позволяет получить в строке s символьное представление числа n. Функция sprintf неконтролирует количество символов, записанное в строку s, поэтому возможно её переполнение.В C99 введена функция snprintf — безопасный вариант sprintf.int snprintf(char s[], size_t n, char format, ...);В строку s будет записано не более чем n - 1 символов согласно спецификации формата,и всегда будет добавлен символ-терминатор строки.
Функция возвращает количество символов, которые были бы записаны в строку s, если её размер был бы неограничен.Следующий фрагмент выполняет конкатенацию двух строк str1 и str2 и помещает результат в строку фиксированного размера buf.int buf[64];snprintf(buf, sizeof(buf), "%s%s", str1, str2);В случае сомнений используйте snprintf!«Обратная» функция sscanf с прототипомint sscanf(char s[], char format, ...);5считывает из форматные данные из строки вместо стандартного потока ввода.
Функция возвращает число успешно считанных полей (как и scanf). Например,sscanf(s, "%d", &n);считывает в переменную n число из строки s. Для sscanf особенно полезен упоминавшийсявыше спецификатор ввода %n. Он позволяет узнать, на каком символе строки остановиласьфункция sscanf.Слово «строка» в русском языке может обозначать два совершенно разных понятия. Вопервых, это просто цепочка символов (string); во вторых, это последовательность символов,занимающая один ряд на экране или в напечатанном тексте (line).
В первом случае иногдаговорят «цепочка».Очень часто бывает так, что стандартный поток ввода ассоциирован либо с терминалом,либо с файлом, который имеет текстовую структуру, то есть разбит на строки (line). В языке Си входной поток является последовательностью символов. Строки текстового файла вовходном потоке разделяются символом ’\n’, не зависимо от того, какой разделитель строктекста в действительности используется в операционной системе (например, в MS-DOSиспользуются два символа ’\r’, ’\n’). Входной поток не имеет специального символапризнака конца входного потока. Таким образом, входной файл видаab bccбудет представлен как поток символов’a’, ’\n’, ’b’, ’ ’, ’b’, ’\n’, ’c’, ’c’, ’\n’, ’\n’Функция gets с прототипомchar * gets(char s[]);считывает одну строку входного текстового файла, то есть последовательность символов досимвола ’\n’, включая его.
Считанные символы помещаются в строку s, причём символ’\n’ заменяется на символ-терминатор ’\0’. В случае, если достигнут конец входногопотока, функция возвращает специальную константу NULL. Обратите внимание, что когдавходной поток ассоциирован с терминалом, для того, чтобы был отмечен конец потока, нужно нажать специальную комбинацию клавиш. Буфер s должен быть достаточного размера,чтобы вместить всю считываемую последовательность символов. Никогда не используйтеэту функцию. Используйте fgets!Функция fgets позволяет считать одну строку входного текстового файла из произвольного потока. Пока мы будем её использовать только для стандартного потока ввода stdin.char * fgets(char s[], int n, FILE *f);Функция считывает из потока не более чем n - 1 символов.
Чтение прерывается, если считано n - 1 символов или достигнут конец строки текста ’\n’. Символ ’\n’ помещаетсяв строку s (отличие от gets!). В любом случае в конец строки s добавляется нулевойбайт-терминатор строки. Пример использования функции:1234unsigned char buf[128];while (fgets(buf, sizeof(buf), stdin)) {printf("%s", buf);}6В этом примере цикл while завершится, как только функция fgets вернёт NULL.Функция puts с прототипомchar *puts(char s[]);добавляет в выходной поток символы из строки s.
Символ-терминатор заменяется на символзавершения строки ’\n’.1.4 Побочный эффект в операциях инкремента и декрементаОсновной эффект операции — это выработка некоторого значения, которое может бытьдалее использовано в выражении. Побочный эффект — это другое воздействие на среду выполнения (например, изменение значения переменной).Нами уже были рассмотрены операции инкремента (увеличения на 1) переменной ++, идекремента (уменьшения на 1) переменной --. В языке Си они существуют в двух формах:префиксной и постфиксной, которые отличаются основным эффектом операции.
Преинкремент ++a — значение переменной вначале увеличивается на 1, и это уже увеличенное значение является значением операции, и далее в выражении может быть использовано. Постинкремент a++ — значением этой операции является значение переменной до увеличения на 1.Например,int a = 5, b, c;b = 5 + ++a;c = 5 + a++;значением переменной b будет 11, а переменной c — 10.1.5 Пример работы со строками текста в файлеРассмотрим такую задачу. На стандартном потоке ввода задаётся последовательностьнеотрицательных целых чисел. Последовательность разбита на строки текста так, что в каждой строке текста входного файла находится хотя бы одно число.
Длина каждой строки текста не превышает 1022 символов. Для каждой строки текста во входном файле найти максимальное целое число, находящееся в этой строке, и напечатать его на стандартный потоквывода. Если строка не содержит чисел (только пробельные символы), на стандартный поток вывода ничего не печатать.Для решения этой задачи мы поступим следующим образом: будем построчно читатьвходной файл с помощью функции fgets, затем анализировать каждую считанную строкус помощью функции scanf. Поскольку по условию задачи требуется анализировать файлстрого по строкам текста, мы проверим, что в результате работы функции fgets строка была считана целиком.123456789#include <stdio.h>#include <string.h>enum { MAX_LEN = 1022 };int main(void){char buf[MAX_LEN + 2];int len, cur, n, max, val;while (fgets(buf, sizeof(buf), stdin)) {710len = strlen(buf);11if (len == sizeof(buf) - 1 && buf[sizeof(buf) - 2] != ’\n’) {12fprintf(stderr, "Line is too long\n");13return 1;14}15cur = 0;16max = -1;17while (sscanf(&buf[cur], "%d%n", &val, &n) == 1) {18if (val < 0) {19fprintf(stderr, "Invalid value\n");20return 1;21}22if (val > max) max = val;23cur += n;24}25n = 0;26sscanf(&buf[cur], " %n", &n);27cur += n;28if (buf[cur]) {29fprintf(stderr, "Invalid string\n");30return 1;31}32if (max >= 0) printf("%d\n", max);33else printf("\n");34}35return 0;36 }Обратите внимание, что запись &buf[cur] в строках 17 и 26 — это подстрока строкиbuf, начинающаяся с позиции cur и продолжающаяся до конца строки buf.
Символ пробела в спецификации формата в строке 26 позволяет пропустить все пробельные символы встроке.1.6 Упражнения1. Написать функцию, которая реверсирует строку, переданную в качестве параметра.2. Написать функцию, которая в массиве из целых чисел все числа, меньшие или равныезаданному переставляет в начало массива, а все числа, большие или равные заданному — в конец массива.3. Проверить на равенство строки (функция strcmp).4. Скопировать из входного потока в выходной все строки, длина которых больше 20.5. Скопировать из входного потока в выходной все строки, которые содержат целое число, большее 2004.8.