Чтение онлайн

на главную

Жанры

Linux программирование в примерах
Шрифт:

В строке 29 устанавливается значение

myname
, а строки 30–45 анализируют опции. Строки 47–51 компилируют регулярное выражение, помещая результаты в
pattern
,
compilе_раttern
увеличивает значение
errors
, если была проблема. (Соединение функций посредством глобальной переменной, как здесь, обычно считается плохой манерой. Для небольших программ, подобным этой, это сойдет, но для более крупных программ такое сопряжение может стать проблемой.) Если не было ошибок, строка 51 увеличивает значение
optind
так, что оставшиеся аргументы
представляют файлы для обработки.

53 if (optind == argc) /* файлов нет, по умолчанию stdin */

54 process("standard input", stdin);

55 else {

56 /* цикл с файлами */

57 for (i = optind; i < argc; i++) {

58 if (strcmp(argv[i], "-") == 0)

59 process("standard input", stdin);

60 else if ((fp = fopen(argv[i], "r")) != NULL) {

61 process(argv[i], fp);

62 fclose(fp);

63 } else {

64 fprintf(stderr, "%s: %s: could not open: %s\n",

65 argv[0], argv[i], strerror(errno));

66 errors++;

67 }

68 }

69 }

70

71 regfree(&pattern);

72 return errors != 0;

73 }

Строки 53–69 обрабатывают файлы, отыскивая соответствующие шаблону строки. Строки 53–54 обрабатывают случай, когда файлы не указаны: программа читает со стандартного ввода. В противном случае, строки 57–68 обрабатывают в цикле файлы. Строка 58 обрабатывает особый случай '

', обозначающий стандартный ввод, строки 60–62 обрабатывают обычные файлы, а строки 63–67 обрабатывают ошибки.

75 /* compile_pattern --- компиляция шаблона */

76

77 void compile_pattern(const char *pat)

78 {

79 int flags = REG_NOSUB; /* информация о месте совпадения не требуется */

80 int ret;

81 #define MSGBUFSIZE 512 /* произвольно */

82 char error[MSGBUFSIZE];

83

84 if (ignore_case)

85 flags |= REG_ICASE;

86 if (extended)

87 flags |= REG_EXTENDED;

88

89 ret = regcomp(&pattern, pat, flags);

90 if (ret != 0) {

91 (void)regerror(ret, &pattern, error, sizeof error);

92 fprintf(stderr, "%s: pattern '%s': %s\n", myname, pat, error);

93 errors++;

94 }

95 }

Строки 75–95 определяют функцию

compile_pattern
.
Она сначала устанавливает
REG_NOSUB
в
flags
, поскольку нам нужно знать лишь «подходит ли строка?», а не «где в строке располагается подходящий текст?»

Строки 84-85 добавляют дополнительные флаги в соответствии с опциями командной строки. Строка 89 компилирует шаблон, а строки 90–94 сообщают о возникших ошибках

97 /* process --- читает строки текста и сопоставляет их с шаблоном */

98

99 void process(const char *name, FILE *fp)

100 {

101 char *buf = NULL;

102 size_t size = 0;

103 char error[MSGBUFSIZE];

104 int ret;

105

106 while (getline(&buf, &size, fp) != -1) {

107 ret = regexec(&pattern, buf, 0, NULL, 0);

108 if (ret != 0) {

109 if (ret != REG_NOMATCH) {

110 (void)regerror(ret, &pattern, error, sizeof error);

111 fprintf(stderr, "%s: file %s: %s\n", myname, name, error);

112 free(buf);

113 errors++;

114 return;

115 }

116 } else

117 printf("%s: %s", name, buf); /* вывести подходящие строки */

118 }

119 free(buf);

120 }

Строки 97–120 определяют функцию

process
, которая читает файл и выполняет сопоставление с регулярным выражением. Внешний цикл (строки 106–119) читает строки ввода. Для избежания проблем с длиной строки мы используем
getline
(см. раздел 3.2.1.9 «Только GLIBC: чтение целых строк:
getline
и
getdelim
»). Строка 107 вызывает
regexec
. Ненулевое возвращаемое значение означает либо неудачное сопоставление, либо какую-нибудь другую ошибку. Строки 109–115 соответственно проверяют
REG_NOMATCН
и выводят ошибку лишь тогда, когда возникла какая-нибудь другая проблема — неудачное сопоставление не является ошибкой

Если возвращаемое значение равно 0, строка совпала с шаблоном и соответственно строка 117 выводит имя файла и совпавшую строку.

122 /* usage --- вывод сообщения об использовании и выход */

123

124 void usage(void)

125 {

126 fprintf(stderr, "usage: %s [-i] [-E] pattern [ files ... ]\n", myname);

127 exit(1);

128 }

Функция

usage
выводит сообщение об использовании и завершает программу. Она вызывается, когда предоставлены недействительные аргументы или не предоставлен шаблон (строки 38–40 и 44–45).

Поделиться:
Популярные книги

Магия чистых душ 2

Шах Ольга
Любовные романы:
любовно-фантастические романы
5.56
рейтинг книги
Магия чистых душ 2

Камень

Минин Станислав
1. Камень
Фантастика:
боевая фантастика
6.80
рейтинг книги
Камень

Прогрессор поневоле

Распопов Дмитрий Викторович
2. Фараон
Фантастика:
попаданцы
альтернативная история
5.00
рейтинг книги
Прогрессор поневоле

Адепт: Обучение. Каникулы [СИ]

Бубела Олег Николаевич
6. Совсем не герой
Фантастика:
фэнтези
попаданцы
9.15
рейтинг книги
Адепт: Обучение. Каникулы [СИ]

Газлайтер. Том 4

Володин Григорий
4. История Телепата
Фантастика:
попаданцы
альтернативная история
аниме
5.00
рейтинг книги
Газлайтер. Том 4

Провинциал. Книга 7

Лопарев Игорь Викторович
7. Провинциал
Фантастика:
боевая фантастика
космическая фантастика
5.00
рейтинг книги
Провинциал. Книга 7

Назад в СССР 5

Дамиров Рафаэль
5. Курсант
Фантастика:
попаданцы
альтернативная история
6.64
рейтинг книги
Назад в СССР 5

Темный Лекарь 3

Токсик Саша
3. Темный Лекарь
Фантастика:
фэнтези
аниме
5.00
рейтинг книги
Темный Лекарь 3

Наследник

Кулаков Алексей Иванович
1. Рюрикова кровь
Фантастика:
научная фантастика
попаданцы
альтернативная история
8.69
рейтинг книги
Наследник

Долг

Кораблев Родион
7. Другая сторона
Фантастика:
боевая фантастика
5.56
рейтинг книги
Долг

Жребий некроманта 2

Решетов Евгений Валерьевич
2. Жребий некроманта
Фантастика:
боевая фантастика
6.87
рейтинг книги
Жребий некроманта 2

Путь (2 книга - 6 книга)

Игнатов Михаил Павлович
Путь
Фантастика:
фэнтези
6.40
рейтинг книги
Путь (2 книга - 6 книга)

Гром над Империей. Часть 1

Машуков Тимур
5. Гром над миром
Фантастика:
фэнтези
5.20
рейтинг книги
Гром над Империей. Часть 1

Великий род

Сай Ярослав
3. Медорфенов
Фантастика:
юмористическое фэнтези
попаданцы
аниме
5.00
рейтинг книги
Великий род