Цикл — это конструкция языка, которая выполняет блок кода больше одного раза.

Мы привыкли, что сценарии выполняются линейно: сверху вниз, строчка за строчкой. Но что делать, если надо повторить какую-нибудь инструкцию несколько раз? Например, как вывести на экран натуральные числа от 1 до 9?

Есть очевидный способ:

<?php
print(1);
print(2);
print(3);
// и так далее...

Но он заставляет писать много кода. И что если требуется вывести последовательность из миллиона чисел? Ещё бывают ситуации, когда заранее неизвестно сколько раз нужно выполнить определённую инструкцию.

Использование циклов значительно упрощает и укорачивает код. Циклы незаменимы в ситуациях, когда заранее неизвестно сколько раз должен выполниться блок кода. Такое число зависит от множества условий и вычисляется в момент выполнения сценария.

Так выглядит цикл в PHP:

<?php
while (<условие цикла>) {
<тело цикла>
}

Использование выражений в циклах

В прошлой главе вы познакомились с понятием выражения и его истинностью.

Выражение часто используется в циклах, оно помещается на место <условие цикла=""> и определяет, будет ли исполнен блок кода — <тело цикла="">.

Если выражение из условия цикла возвращает истину, то выполнение сразу перейдёт к блоку «тело цикла». Если же оно вернёт ложь, то тело цикла не будет исполнено. Сценарий продолжит выполняться, как обычно, со следующей строки после цикла.

Название «циклы» отражает суть их действия — они «зацикливают» линейное исполнение на своём блоке кода и не дают сценарию выполняться дальше, пока условие цикла будет истинным.

👉 Последовательность исполнения кода при использовании циклов:

  1. Обычное исполнение кода, строчка за строчкой, пока не встретился цикл.

  2. Встретился цикл: выполняем условие цикла.

  3. Если условие вернуло ложь: выходим из цикла, выполняем строчку после цикла и продолжаем линейное исполнение.

  4. Если условие вернуло истину: выполняем всё тело цикла.

  5. Повторяем пункт 2.

Каждая последовательность из шагов 2 — 4, то есть очередное выполнение кода в теле цикла — называется итерацией.

Количество итераций должно быть конечным, бесконечное выполнение одного блока кода не входит в наши планы.

А значит, необходимо заранее предусмотреть ситуацию, когда истинное условие станет ложным.

Теперь вернёмся к задаче по выводу на экран всех натуральных чисел:

<?php
$last_num = 1;

while ($last_num < 10) {
    print($last_num);
    $last_num = $last_num + 1;
}

Этот цикл в своём теле содержит две инструкции. Первая выводит на экран цифру из переменной. Вторая инструкция увеличивает значение переменной на единицу. Сколько раз будет исполнен такой цикл?

Циклы выполняются, пока их условие остаётся истинным, а в нашем условии значение переменной должно быть меньше десяти. Так как начальное значение переменной — единица, то несложно посчитать, что цикл выполнится ровно девять раз. На десятый раз значение переменной $last_num станет равно десяти и условие $last_num < 10 перестанет быть истинным.

Циклы и массивы

Чаще всего циклы используются для работы с массивами — для перечисления всех элементов массива и выполнения какого-нибудь действия с каждым из этих элементов. Умение использовать циклы и массивы совместно позволяет выполнять множество интересных и разнообразных задач!

Мы уже научились работать с массивами. Например, мы можем показать все значения массива, обратившись к каждому из элементов по его индексу. Но это утомительно: обращаться к каждому из элементов массива по очереди, когда мы хотим показать все его значения.

С помощью циклов показывают содержимое любого массива, и это требует всего несколько строк кода!

Перепишем пример с выводом списка любимых сериалов, задействовав цикл:

<?php
$fav_shows = ["game of thrones", "american horror storry", "walking dead"];
$cur_index = 0;

while ($cur_index < 3) {
    print($fav_shows[$cur_index] . ", ");
    $cur_index = $cur_index + 1;
}

В этом примере цикл выводит элемент по индексу. Индекс теперь находится в переменной $cur_index и начальное значение у него ноль. Значение переменной увеличивается на единицу с каждой итерацией цикла, пока не достигнет трёх. В этот момент условие $cur_index < 3 перестанет быть истинным и цикл остановится, перебрав весь массив.

Потренируйтесь использовать циклы с массивами, пройдя этот тренажёр.

foreach — специальный цикл для массивов

Циклы в PHP могут быть разных типов. Выше мы познакомились с циклом типа while. Его главная особенность — необходимо указывать выражение в его условии. Но while не единственный вид циклов в PHP. Есть ещё как минимум два других.

Массивы и циклы так часто используются вместе, что разработчики языка даже добавили вид цикла специально для перебора массивов. Цикл называется foreach. Зачем понадобилось придумывать этот цикл?

Дело в том, что while слишком универсален. А платой за эту универсальность всегда будет более сложный и объёмный код. Приходится придумывать условие, следить, чтобы оно не было бесконечным. А в теле цикла обязательно надо не забыть увеличивать переменную-счётчик. И всё это нужно для простого перебора элементов массива. Неужели нельзя сделать проще?

👉  foreach решает все эти проблемы. Его возможности:

  • не требуется писать условие;
  • позволяет получать ключи массива;
  • сам присваивает очередной элемент массива переменной.

Цикл foreach незаменим, когда дело доходит до итерации по ассоциативным массивам. Например: у нас есть данные пользователя, которые хранятся в ассоциативном массиве. На сайте понадобилось сделать страницу с информацией об этом юзере. Задача в том, чтобы показать на странице все данные, которые известны об этом человеке. Выглядеть это должно так:

Имя: Евгений
Возраст: 27
Род занятий: Программист

Оригинальный массив, который надо показать в таком виде:

$user = [
    'Имя' => 'Евгений',
    'Возраст' => '27',
    'Род занятий' => 'Программист'
];

Код сценария, который обойдёт массив и покажет всё его содержимое, займёт всего четыре строчки:

foreach ($user as $key => $value) {
    print($key . ': ');
    print($value . '<br>');
}

На каждой итерации цикла внутри его тела будут определяться переменные $key и $value. В первую будет попадать очередной ключ массива, а во вторую — очередное значение по этому ключу.