Выбрать главу

Однако накопленный опыт часто заставляет нас искать в Arduino то, чем мы привыкли пользоваться в повседневной работе. При переходе из мира больших систем в миниатюрный мир Arduino первое, что сразу бросается в глаза, — это простота разработки программ для Arduino. Приступать к созданию большой системы без использования приемов разработки через тестирование, системы управления версиями и процедуры гибкой разработки было бы слишком опрометчиво. В то же время большой проект для Arduino может состоять всего из 200 строк кода, написанных одним человеком. Если этот человек — опытный программист, он просто будет хранить все тонкости в памяти, не нуждаясь в привлечении инструментов, обычных в большой разработке.

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

Почему вам не нужны потоки выполнения

Если вам так много лет, что вы застали времена, когда большое распространение имели домашние компьютеры, программируемые на Бейсике, вы должны помнить, что компьютеры «в каждый момент времени делают что-то одно». Если игре, написанной на Бейсике, требовалось одновременно перемещать несколько спрайтов, вам приходилось прибегать к уловке с общим циклом, в котором каждый спрайт перемещался на небольшое расстояние.

Этот образ мышления отлично подходит для программирования Arduino. Вместо создания множества потоков выполнения, каждый из которых отвечает за перемещение единственного спрайта, достаточно единственного потока, перемещающего спрайты по очереди небольшими шагами, не блокируя ничто другое.

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

Функции setup и loop

В каждом скетче требуется реализовать две функции, setup и loop, и такой подход выбран не случайно. Фактически функция loop вызывается снова и снова, и именно по этой причине ее работа не должна блокироваться. Код в функции loop должен действовать виртуозно, чтобы она выполнялась моментально и тут же запускалась вновь.

Оценка, затем действие

Большинство проектов для Arduino предназначено для управления чем-то. Поэтому функция loop часто:

• проверяет нажатие кнопки или превышение данных с некоторого датчика порогового значения;

• выполняет соответствующее действие.

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

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

// sketch_14_01_flashing_1

const int ledPin = 13;

const int switchPin = 5;

const int period = 1000;

boolean flashing = false;

void setup()

{

  pinMode(ledPin, OUTPUT);

  pinMode(switchPin, INPUT_PULLUP);

}

void loop()

{

  if (digitalRead(switchPin) == LOW)

  {

    flashing = ! flashing;

  }

  if (flashing)

  {

    digitalWrite(ledPin, HIGH);

    delay(period);

    digitalWrite(ledPin, LOW);

    delay(period);

  }

}

Проблема данной реализации в том, что она проверяет нажатие кнопки только после того, как завершится цикл включения/выключения светодиода. Если кнопка будет нажата во время этого цикла, факт нажатия зафиксирован не будет. Это может быть не важно для нормальной работы скетча, но если важно фиксировать каждое нажатие кнопки, следует полностью исключить любые задержки в функции loop. Фактически после перехода в режим мигания Arduino будет тратить основное время на задержки и только малую часть времени — на проверку состояния кнопки.

Пример в следующем разделе решает эту проблему.

Пауза без приостановки

Предыдущий скетч можно переписать без использования функции delay:

// sketch_14_02_flashing_2

const int ledPin = 13;

const int switchPin = 5;

const int period = 1000;

boolean flashing = false;

long lastChangeTime = 0;

int ledState = LOW;

void setup()

{

  pinMode(ledPin, OUTPUT);

  pinMode(switchPin, INPUT_PULLUP);

}

void loop()

{

  if (digitalRead(switchPin) == LOW)

  {

    flashing = ! flashing;

    // и выключить светодиод

    if (! flashing)