Как мы говорили в главе 1, в циклах whileи forпроверка условия окончания цикла выполняется наверху. В Си имеется еще один вид цикла, do-while, в котором эта проверка в отличие от whileи forделается внизу после каждого прохождения тела цикла, т. е. после того, как тело выполнится хотя бы один раз. Цикл do-whileимеет следующий синтаксис:
do
инструкция
while ( выражение );
Сначала выполняется инструкция , затем вычисляется выражение . Если оно истинно, то инструкция выполняется снова и т. д. Когда выражение становится ложным, цикл заканчивает работу. Цикл do-whileэквивалентен циклу repeat-untilв Паскале с той лишь разницей, что в первом случае указывается условие продолжения цикла, а во втором - условие его окончания.
Опыт показывает, что цикл do-whileиспользуется гораздо реже, чем whileи for. Тем не менее потребность в нем время от времени возникает, как, например, в функции itoa (обратной по отношению к atoi ), преобразующей число в строку символов. Выполнить такое преобразование оказалось несколько более сложным делом, чем ожидалось, поскольку простые алгоритмы генерируют цифры в обратном порядке. Мы остановились на варианте, в котором сначала формируется обратная последовательность цифр, а затем она реверсируется.
/* itoa: преобразование n в строку s */
void itoa(int n, char s[])
{
int i, sign;
if ((sign = n) ‹ 0) /* сохраняем знак */
n =-n; /* делаем n положительным */
i = 0;
do {/* генерируем цифры в обратном порядке */
s[i++] = n %10 + '0'; /* следующая цифра */
} while ((n /= 10) › 0); /* исключить ее */
if (sign ‹ 0)
s[i++] = '-';
s[i] = '\0';
reverse(s);
}
Конструкция do-whileздесь необходима или по крайней мере удобна, поскольку в s посылается хотя бы один символ, даже если n равно нулю. В теле цикла одну инструкцию мы выделили фигурными скобками (хотя они и избыточны), чтобы неискушенный читатель не принял по ошибке слово whileза начало цикла while.
Упражнение 3.4. При условии, что для представления чисел используется дополнительный код, наша версия itoa не справляется с самым большим по модулю отрицательным числом, значение которого равняется -(2 n-1), где n - размер слова. Объясните, чем это вызвано. Модифицируйте программу таким образом, чтобы она давала правильное значение указанного числа независимо от машины, на которой выполняется.
Упражнение 3.5. Напишите функцию itob(n,s,b), которая переводит целое n в строку s, представляющую число по основанию b. В частности, itob(n, s, 16) помещает в s текст числа n в шестнадцатеричном виде.
Упражнение 3.6. Напишите версию itoa с дополнительным третьим аргументом, задающим минимальную ширину поля. При необходимости преобразованное число должно слева дополняться пробелами.
3.7 Инструкции break и continue
Иногда бывает удобно выйти из цикла не по результату проверки, осуществляемой в начале или в конце цикла, а каким-то другим способом. Такую возможность для циклов for, whileи do-while, а также для переключателя switchпредоставляет инструкция break. Эта инструкция вызывает немедленный выход из самого внутреннего из объемлющих ее циклов или переключателей.
Следующая функция, trim , удаляет из строки завершающие пробелы, табуляции, символы новой строки; breakиспользуется в ней для выхода из цикла по первому обнаруженному справа символу, отличному от названных.
/* trim: удаляет завершающие пробелы, табуляции и новые строки */
int trim(char s[])
{
int n;
for (n = strlen(s)-1; n ›= 0, n--)
if (s[n]!= ' '&& s[n]!= '\t'&& s[n]!= '\n')
break;
s[n+1] = '\0';
return n;
}
С помощью функции strlen можно получить длину строки. Цикл forпросматривает его в обратном порядке, начиная с конца, до тех пор, пока не встретится символ, отличный от пробела, табуляции и новой строки. Цикл прерывается, как только такой символ обнаружится или n станет отрицательным (т. е. вся строка будет просмотрена). Убедитесь, что функция ведет себя правильно и в случаях, когда строка пуста или состоит только из символов-разделителей.
Инструкция continueв чем-то похожа на break, но применяется гораздо реже. Она вынуждает ближайший объемлющий ее цикл ( for, whileили do-while) начать следующий шаг итерации. Для whileи do-whileэто означает немедленный переход к проверке условия, а для for- к приращению шага. Инструкцию continueможно применять только к циклам, но не к switch. Внутри переключателя switch, расположенного в цикле, она вызовет переход к следующей итерации этого цикла.
Читать дальше