Учебник по "программированию на Arduino"

Лекция №6. Циклы

Цикл for, в простонародии счётчик. При запуске принимает три настройки: инициализация, условие и изменение. Цикл for обычно содержит переменную, которая изменяется на протяжении работы цикла, а мы можем пользоваться её значением в своих целях.

Инициализация - здесь обычно присваивают начальное значение переменной цикла. Например: int i = 0;

Условие - здесь задаётся условие, при котором выполняется цикл. Как только условие нарушается, цикл завершает работу. Например: i < 100;

Изменение - здесь указывается изменение переменной цикла на каждой итерации. Например: i++;

Пример:

for (int i = 0; i < 100; i++) {

// тело цикла

}

В теле цикла мы можем пользоваться значением переменной i, которая примет значения от 0 до 99 на протяжении работы цикла, после этого цикл завершается. Как это использовать? Вспомним предыдущий урок про массивы и рассмотрим пример:

// создадим массив на 100 ячеек

byte myArray[100];

for (int i = 0; i < 100; i++) {

// заполним все ячейки значением 25

myArray[i] = 25;

}

Именно при помощи цикла for очень часто работают с массивами. Можно сложить все элементы массива для поиска среднего арифметического:

// создадим массив

byte myVals[] = {5, 60, 214, 36, 98, 156};

// переменная для хранения суммы

int sum = 0;

for (int i = 0; i < 6; i++) {

// суммируем весь массив в sum

sum += myVals[i];

}

// разделим sum на количество элементов

// и получим среднее арифметическое!

sum /= 6;

// получилось 94!

Что касается особенностей использования for в языке C++: любая его настройка является необязательной, то есть её можно не указывать для каких-то особенных алгоритмов. Например вы не хотите создавать переменную цикла, а использовать для этого другую имеющуюся переменную.

int index = 0; // свой счётчик

for (; index < 60; index += 10) {

// переменная index принимает значения

// 0, 10, 20, 30, 40, 50

}

В цикле for можно сделать несколько счётчиков, несколько условий и несколько инкрементов, разделяя их при помощи оператора запятая , смотрите пример:

// объявить i и j

// прибавлять i+1 и j+2

for (byte i = 0, j = 0; i < 10; i++, j += 2) {

// тут i меняется от 0 до 9

// и j меняется от 0 до 18

}

Также в цикле может вообще не быть настроек, и такой цикл можно считать вечным:

for (;;) {

// выполняется вечно...

}

Использование замкнутых циклов не очень приветствуется, но иногда является очень удобным способом поймать какое-то значение, или дать программе "зависнуть" при наступлении ошибки. Выйти из цикла при помощи оператора break, о нём поговорим чуть ниже.

Цикл "for each"

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

int vals[] = {10, 11, 12, 13, 14};

for (int i = 0; i < sizeof(vals); i++) {

Serial.println(vals[i]);

}

Как это работает: мы завели цикл for с переменной-счётчиком i, который меняется от 0 до размера массива, который вы вычисляем через sizeof(). Внутри цикла мы используем счётчик как индекс массива, чтобы обратиться к каждой его ячейке как [i]. Но цикл for для работы с массивом можно записать более красиво:

for (тип_данных_массива переменная : массив) {}

В нашем примере вывода это будет выглядеть так:

int vals[] = {10, 11, 12, 13, 14};

for (int val : vals) {

Serial.println(val);

}

Как это работает: мы создаём переменную val такого же типа как массив, а также указываем имя массива через двоеточие. На каждой итерации цикла переменная val будет принимать значение ячейки массива в порядке от 0 до размера массива с шагом 1. Таким образом мы решили ту же задачу, но написали меньше кода.

Важный момент: на каждой итерации цикла значение ячейки присваивается к переменной val, то есть фактически мы можем только прочитать значение (через буферную переменную). Для непосредственного доступа к элементам массива нужно создавать ссылку, то есть просто добавить оператор &

int vals[] = {10, 11, 12, 13, 14};

for (int &val : vals) {

Serial.println(val);

val = 0;

}

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

В такой реализации цикла for у нас нет счётчика, что может быть неудобным для некоторых алгоритмов, но счётчик всегда можно добавить свой. Например забьём массив числами от 0 до 90 с шагом 10:

int vals[10];

int count = 0;

for (int &val : vals) {

val = count * 10;

count++;

}

И это будет всё ещё компактнее классического for.

Оператор break

Оператор break (англ. "ломать") позволяет выйти из цикла. Использовать его можно как по условию, так и как-угодно-удобно. Например давайте досрочно выйдем из цикла при достижении какого-то значения:

for (int i = 0; i < 100; i++) {

// покинуть цикл при достижении 50

if (i == 50) break;

}

Или вот такой абстрактный пример, покинем "вечный" цикл при нажатии на кнопку:

for (;;) {

if (кнопка нажата) break;

}

Выход из цикла является не единственным интересным инструментом, ещё есть оператор пропуска - continue.

Оператор continue

Оператор continue (англ. "продолжить") досрочно завершает текущую итерацию цикла и переходит к следующей. Например давайте заполним массив, как делали выше, но пропустим один элемент:

// создадим массив на 100 ячеек

byte myArray[100];

for (int i = 0; i < 100; i++) {

// если i равно 10, пропускаем

if (i == 10) continue;

// заполним все ячейки значением 25

myArray[i] = 25;

}

Таким образом элемент под номером 10 не получит значения 25, итерация завершится до операции присваивания.

Цикл while

Цикл while (англ. "пока"), он же "цикл с предусловием", выполняется до тех пор, пока верно указанное условие. Если условие изначально неверно, цикл будет пропущен, не сделает ни одной итерации. Объявляется очень просто: ключевое слово while, далее условие в скобках, и вот уже тело цикла:

int i = 0;

while (i < 10) {

i++;

}

Это полный аналог цикла for с настройками (int i = 0; i < 10; i++) . Единственное отличие в том, что на последней итерации i примет значение 10, так как на значении 9 цикл разрешит выполнение. Ещё интересный вариант, работает на основе того факта, что любое число кроме нуля обрабатывается логикой как true:

byte a = 5;

while (a--) {

// выполнится 5 раз

}

Цикл while тоже удобно использовать как вечный цикл, например, ожидая наступление какого-либо события (нажатие кнопки):

// выполняется, пока не нажата кнопка

while (кнопка не нажата);

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

while (true);

Всё, вертимся здесь бесконечно! Помимо цикла с предусловием есть ещё цикл с постусловием, так называемый do while

Цикл do while

do while - "делать пока", работа этого цикла полностью аналогична циклу while за тем исключением, что здесь условие задаётся после цикла, т.е. цикл выполнится как минимум один раз, затем проверит условие, а не наоборот. Пример:

do {

// тело цикла

} while (условие);

Web hosting by Somee.com