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

на главную - закладки

Жанры

Linux программирование в примерах

Роббинс Арнольд

Шрифт:

63 printf("Finishing dir: %s\n", finish);

64 }

65

66 return (errors != 0);

67 }

Строки 3–11 включают заголовочные файлы. По крайней мере в GLIBC 2.3.2 перед включением любого заголовочного файла необходимы

#define
для
_XOPEN_SOURCE
и
_XOPEN_SOURCE_EXTENDED
. Они дают возможность получить объявления и значения флагов, которые
nftw
предоставляет свыше предоставляемых
ftw
. Это специфично для GLIBC. Потребность в этом в конечном счете исчезнет, когда GLIBC станет полностью совместимой со стандартом POSIX 2001.

Строки 35–44 обрабатывают опции. Опция

– с
добавляет к флагам
nftw FTW_CHDIR
. Это эксперимент с целью увидеть, сможете ли вы оказаться где-то в другом месте от того, где начинали. Кажется, это возможно, если
nftw
завершается неудачей, в противном случае вы заканчиваете там же, где начинали. (POSIX не документирует это явным образом, но целью, похоже, было действительно заканчивать там же, где начинали. Стандарт не говорит, что функция обратного вызова не должна менять текущий каталог.)

Строка 49 сохраняет начальный каталог для дальнейшего использования, пользуясь

getcwd
.

Строка 51 вычисляет число дескрипторов, которые может использовать

nftw
. Мы не хотим, чтобы она использовала все доступные дескрипторы файлов, если функция обратного вызова также хочет открывать файлы. В вычислении используется
getdtablesize
(см. раздел 4.4.1 «Понятие о дескрипторах файлов») для получения максимально возможного числа и вычета из него
SPARE_FDS
, который был вычислен ранее в строке 13.

Эта процедура служит основанием для больших объяснений. В обычном случае по крайней мере три дескриптора уже используются для стандартного ввода, стандартного вывода и стандартной ошибки.

nftw
нужно некоторое количество дескрипторов файлов для открытия и чтения каталогов; внутри себя
opendir
использует
open
при открытии каталогов для чтения. Если функции обратного вызова также нужно открывать файлы, мы должны предотвратить израсходование функцией
nftw
всех доступных дескрипторов файлов для открывания каталогов. Мы делаем это, вычитая некоторое число из максимально допустимого. Для данного примера мы выбрали пять, но если функции обратного вызова нужно открывать файлы, должно использоваться большее число, (
nftw
знает, как восстановиться при израсходовании дескрипторов файлов; мы не должны беспокоиться о таком случае.)

Строки 52–58 являются главным циклом над нашими аргументами; строки 53–57 проверяют ошибки; когда они появляются, код выводит диагностическое сообщение и увеличивает значение переменной

errors
.

Строки 60–64 являются частью эксперимента с

FTW_CHDIR
, выводящего начальный и конечный каталоги, если было использовано
– с
.

По-настоящему интересной функцией является

process
; это функция обратного вызова, которая обрабатывает каждый файл. Она использует базовый шаблон для функции обратного вызова
nftw
, который является оператором
switch
для значения
flag
.

69 /* process --- выводит каждый файл на нужном уровне */

70

71 int process(const char "file, const struct stat *sb,

72 int flag, struct FTW *s)

73 {

74 int retval = 0;

75 const char *name = file + s->base;

76

77 printf("%*s", s->level * 4, ""); /* сделать отступ */

78

79 switch (flag) {

80 case FTW_F:

81 printf("%s (file)\n", name);

82 break;

83 case FTW_D:

84 printf("%s (directory)\n", name);

85 break;

86 case FTW_DNR:

87 printf("%s (unreadable directory)\n", name);

88 break;

89 case FTW_SL:

90 printf("%s (symbolic link)\n", name);

91 break;

92 case FTW_NS:

93 printf("%s (stat failed): %s\n", name, strerror(errno));

94 break;

95 case FTW_DP:

96 case FTW_SLN:

97 printf("%s: FTW_DP or FTW_SLN: can't happen'\n", name);

98 retval = 1;

99 break;

100 default:

101 printf("%s: unknown flag %d: can't happen'\n", name, flag);

102 retval = 1;

103 break;

104 }

105

106 return retval;

107 }

Строка 75 использует '

file + s->base
' для получения имени из полного пути. Это значение указателя сохраняется в переменной
name
для повторного использования в функции.

Строка 77 делает отступ нужного размера, используя красивый трюк. Используя

%*s
,
printf
получает от первого аргумента ширину поля. Это вычисляется динамически как '
level * 4
'. Строка, которая должна быть выведена — «», пустая строка. Конечным результатом является то, что
printf
создает для нас отступ нужного размера без необходимости запуска цикла.

Строки 79–104 являются оператором

switch
. В данном случае он не делает ничего весьма интересного, кроме вывода имени файла и его типа (файл, каталог и т.д.)

Хотя эта программа не использует

struct stat
, должно быть ясно, что вы могли бы сделать в функции обратного вызова все, что хотите.

ЗАМЕЧАНИЕ. Джим Мейеринг (Jim Meyering), сопроводитель GNU Coreutils, замечает, что дизайн

nftw
несовершенен из-за ее рекурсивной природы. (Она рекурсивно вызывает себя при обработке подкаталогов.) Если иерархия каталогов становится действительно глубокой, в диапазоне уровней 20 000–40 000 (!),
nftw
может выйти за пределы размера стека, уничтожив программу. Есть также и другие проблемы, связанные с дизайном
nftw
. Версия GNU Coreutils после 5.0 исправляет это путем использования набора процедур
fts
(см. fts(3)).

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

Мессия

Старлинг Борис
121. Книга-загадка, книга-бестселлер
Детективы:
триллеры
маньяки
7.23
рейтинг книги
Мессия

Граф

Ланцов Михаил Алексеевич
6. Помещик
Фантастика:
альтернативная история
5.00
рейтинг книги
Граф

Звездная Кровь. Экзарх II

Рокотов Алексей
2. Экзарх
Старинная литература:
прочая старинная литература
5.00
рейтинг книги
Звездная Кровь. Экзарх II

Хозяин Теней 6

Петров Максим Николаевич
6. Безбожник
Фантастика:
аниме
фэнтези
фантастика: прочее
попаданцы
5.00
рейтинг книги
Хозяин Теней 6

На границе империй. Том 10. Часть 10

INDIGO
Вселенная EVE Online
Фантастика:
космическая фантастика
попаданцы
5.00
рейтинг книги
На границе империй. Том 10. Часть 10

Наследник

Назимов Константин Геннадьевич
3. Травник
Фантастика:
фэнтези
6.80
рейтинг книги
Наследник

Практик

Листратов Валерий
5. Ушедший Род
Фантастика:
аниме
фэнтези
попаданцы
5.00
рейтинг книги
Практик

Бастард Императора. Том 5

Орлов Андрей Юрьевич
5. Бастард Императора
Фантастика:
попаданцы
аниме
фэнтези
5.00
рейтинг книги
Бастард Императора. Том 5

Эволюционер из трущоб

Панарин Антон
1. Эволюционер из трущоб
Фантастика:
попаданцы
аниме
фэнтези
фантастика: прочее
5.00
рейтинг книги
Эволюционер из трущоб

Сталин

Радзинский Эдвард Станиславович
3. Загадки жизни и смерти
Проза:
историческая проза
7.36
рейтинг книги
Сталин

Вернуть невесту. Ловушка для попаданки

Ардова Алиса
1. Вернуть невесту
Любовные романы:
любовно-фантастические романы
8.49
рейтинг книги
Вернуть невесту. Ловушка для попаданки

Тринадцатый V

NikL
5. Видящий смерть
Фантастика:
фэнтези
попаданцы
аниме
5.00
рейтинг книги
Тринадцатый V

Начало

Харенко Алина Александровна
1. Второй шанс
Фантастика:
фэнтези
5.00
рейтинг книги
Начало

Цикл "Отмороженный". Компиляция. Книги 1-14

Гарцевич Евгений Александрович
Отмороженный
Фантастика:
боевая фантастика
рпг
постапокалипсис
5.00
рейтинг книги
Цикл Отмороженный. Компиляция. Книги 1-14