Обратите внимание: для указания на отсутствие инициализирующего оператора точка с запятой необходима, точнее, точка с запятой представляет пустой инициализирующий оператор. В этом цикле for
тело также пусто, поскольку все его действия осуществляются в условии и выражении. Условие решает, когда придет время прекращать просмотр, а выражение увеличивает итератор.
Отсутствие части
эквивалентно расположению в условии значения true
. Поскольку условие истинно всегда, тело цикла for
должно содержать оператор, обеспечивающий выход из цикла. В противном случае цикл будет выполняться бесконечно.
for (int i = 0; /*
//
}
В заголовке for
может также отсутствовать
. В таких циклах либо условие, либо тело должно делать нечто обеспечивающее итерацию. В качестве примера перепишем цикл while
, читающий ввод в вектор целых чисел.
vector
for (int i; cin >> i; /*
v.push_back(i);
В этом цикле нет никакой необходимости в выражении, поскольку условие изменяет значение переменной i
. Условие проверяет входной поток, поэтому цикл заканчивается, когда прочитан весь ввод или произошла ошибка ввода.
Упражнение 5.15. Объясните каждый из следующих циклов. Исправьте все обнаруженные ошибки.
(a) for (int ix = 0; ix != sz; ++ix) { /* ... */ }
if (ix != sz)
// ...
(b) int ix;
for (ix != sz; ++ix) { /* ... */ }
(c) for (int ix = 0; ix != sz; ++ix, ++sz) { /* ... */ }
Упражнение 5.16. Цикл while
особенно хорош, когда необходимо выполнить некое условие; например, когда нужно читать значения до конца файла. Цикл for
считают циклом пошагового выполнения: индекс проходит диапазон значений в коллекции. Напишите идиоматическое использование каждого цикла, а затем перепишите каждый случаи использования в другой конструкции цикла. Если бы вы могли использовать только один цикл, то какой бы вы выбрали и почему?
Упражнение 5.17. Предположим, есть два вектора целых чисел. Напишите программу, определяющую, не является ли один вектор префиксом другого. Для векторов неравной длины сравнивайте количество элементов меньшего вектора. Например, если векторы содержат значения 0, 1, 1, 2
и 0, 1, 1, 2, 3, 5, 8
соответственно, ваша программа должна возвратить true
.
for
for
, который перебирает элементы контейнера или другой последовательности. Синтаксис for
(range for
) таков:
for (
vector
или string
, у которого есть функции-члены begin()
и end()
, возвращающие итераторы (см. раздел 3.4).
определяет переменную. Каждый элемент последовательности должен допускать преобразование в тип переменной (см. раздел 4.11). Проще всего гарантировать соответствие типов за счет использования спецификатора типа auto
(см. раздел 2.5.2). Так компилятор выведет тип сам. Если необходима запись в элементы последовательности, то переменная цикла должна иметь ссылочный тип.
На каждой итерации управляющая переменная определяется и инициализируется следующим значением последовательности, а затем выполняется
. Как обычно,
может быть одиночным оператором или блоком. Выполнение завершается, когда все элементы обработаны.
Несколько таких циклов уже было представлено, но для завершенности рассмотрим цикл, удваивающий значение каждого элемента в векторе:
vector
//
for (auto &r : v) //
r *= 2; //
Заголовок for
объявляет, что управляющая переменная цикла r
связана с вектором v
. Чтобы позволить компилятору самостоятельно вывести тип переменной r
, используем спецификатор auto
. Поскольку предполагается изменение значений элементов вектора v
, объявим переменную r
как ссылку. При присвоении ей значений в цикле фактически присваивается значение элементу, с которым связана переменная r
в данный момент.