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

ЖАНРЫ

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

106 }

107 else if ((size_t)info->level > prev_level)

108 {

109 /* Нисхождение по иерархии.

110 Очистить аккумуляторы для *всех* уровней между prev_level

111 и текущим. Глубина может значительно меняться,

112 например, от 1 до 10. */

113 int i;

114 for (i = prev_level +1; i <= info->level; i++)

115 sum_ent[i] = sum_subdir[i] = 0;

116 }

117 else /* info->level < prev_level */

118 {

119 /*
Восхождение по иерархии.

120 nftw обрабатывает каталог лишь после всех элементов,

121 в которых был обработан каталог. Когда глубина уменьшается,

122 передать суммы от детей (prev_level) родителям.

123 Здесь текущий уровень всегда меньше, чем

124 предыдущий. */

125 assert (<size_t) info->level == prev_level - 1);

126 size_to_print += sum_ent[prev_level];

127 if (!opt_separate_dirs)

128 size_to_print += sum_subdir[prev_level];

129 sum_subdir[info->level] += (sum_ent[prev_level]

130 + sum_subdir[prev_level]);

131 }

132 }

Строки 101–132 сравнивают текущий уровень с предыдущим. Возможны три случая.

Уровни те же самые

В этом случае нет необходимости беспокоиться о статистике дочерних элементов. (Строки 103–106.)

Текущий уровень выше предыдущего

В этом случае мы спустились по иерархии, и статистику нужно восстановить (строки 107–116). Термин «аккумулятор» в комментарии подходящий: каждый элемент аккумулирует общее дисковое пространство, использованное на этом уровне. (На заре вычислительной техники регистры центрального процессора часто назывались «аккумуляторами».)

Текущий уровень ниже предыдущего

В этом случае мы завершили обработку всех дочерних элементов каталога и только что вернулись обратно в родительский каталог (строки 117–131). Код обновляет суммы, включая

size_to_print
.

134 prev_level = info->level; /* Установить статические переменные */

135 first_call = 0;

136

137 /* Включить элемент каталога в общую сумму для содержащего

138 каталога, если не указана --separate-dirs (-S). */

139 if (!(opt_separate_dirs && IS_FTW_DIR_TYPE(file_type)))

140 sum_ent[info->level] += size;

141

142 /* Даже если каталог нельзя прочесть или перейти в него,

143 включить его размер в общую сумму, ... */

144 tot_size += size;

145

146 /* ...но не выводить для него итог, поскольку без размера(-ов)

147
потенциальных элементов, это может сильно запутывать. */

148 if (file_type == FTW_DNR || file_type == FTW_DCH)

149 return 0;

150

151 /* Если мы не считаем элемент, например, потому что это прямая

152 ссылка на файл, который уже посчитан (и --count-links), не

153 выводить для него строку. */

154 if (!print)

155 return 0;

Строки 134–135 устанавливают статические переменные

prev_level
и
first_call
таким образом, что они содержат правильные значения для последующего вызова
process_file
, гарантируя, что весь предыдущий код работает правильно.

Строки 137–144 выверяют статистику на основе опций и типа файла. Комментарии и код достаточно просты. Строки 146–155 сразу завершают функцию, если сведения не должны выводиться.

157 /* FIXME: Это выглядит подозрительно годным для упрощения. */

158 if ((IS_FTW_DIR_TYPE(file_type) &&

159 (info->level <= max_depth || info->level == 0))

160 || <(opt_all && info->level <= max_depth) || info->level == 0))

161 {

162 print_only_size(size_to_print);

163 fputc('\t', stdout);

164 if (arg_length)

165 {

166 /* Вывести имя файла, но без суффикса каталога '.' или '/.'

167 который мы, возможно, добавили в main. */

168 /* Вывести все до добавленной нами части. */

169 fwrite(file, arg_length, 1, stdout);

170 /* Вывести все после добавленного нами. */

171 fputs(file + arg_length + suffix_length

172 + (file[arg_length + suffix_length] == '/'), stdout);

173 }

174 else

175 {

176 fputs(file, stdout);

177 }

178 fputc('\n', stdout);

179 fflush(stdout);

180 }

181

182 return 0;

183 }

Условие в строках 158–160 сбивает с толку, и комментарий в строке 157 указывает на это. Условие утверждает: «Если (1a) файл является каталогом и (1b) уровень меньше максимального для вывода (переменные —

– max-depth
и
max_depth
) или нулевой, или (2a) должны быть выведены все файлы и уровень меньше, чем максимальный для вывода, или (2b) уровень нулевой», тогда вывести файл. (Версия
du
после 5.0 использует в этом случае несколько менее запутанное условие.)

Поделиться с друзьями: