Изучаем Arduino: инструметы и методы технического волшебства - [12]

Шрифт
Интервал

1. Сохраняем предыдущее и текущее состояния кнопки (при инициализации LOW).

2. Считываем текущее состояние кнопки.

3. Если текущее состояние кнопки отличается от предыдущего, ждем 5 мс, потому что кнопка, возможно, изменит свое состояние.

4. Подождав 5 мс, считываем состояние кнопки и делаем его текущим состоянием кнопки.

5. Если предыдущее состояние кнопки было LOW, а текущее - HIGH, переключаем состояние светодиода.

- 56 -

6. Устанавливаем предыдущее состояние кнопки в качестве текущего.

7. Возвращаемся к шагу 2.

Данный алгоритм - прекрасный пример для изучения функций. Функция - это оператор, который может принимать входные аргументы, выполнять фрагмент кода с их использованием и, возможно, возвращать результат. Не зная этого, вы уже встречали функции в программах. Например, digitalWrite() - это функция, которая принимает в качестве аргументов номер контакта и значение ( HIGH или LOW), и устанавливает это значение на контакте. Чтобы упростить программу, можно определить свои собственные функции для инкапсуляции действий, которые придется повторять неоднократно.

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

Листинг 2.5. Подавление дребезга кнопки — debounce.ino

const int LED=9; // Контакт 9 для подключения светодиода

const int BUTTON=2; // Контакт 2 для подключения кнопки

boolean lastButton = LOW;// Переменная для сохранения предыдущего

// состояния кнопки

boolean currentButton = LOW;// Переменная для сохранения текущего

// состояния кнопки

boolean ledOn = false;// Текущее состояние светодиода

//(включен/выключен)


void setup()

{

pinMode (LED, OUTPUT);// Сконфигурировать контакт светодиода как выход

pinMode (BUTTON, INPUT);//Сконфигурировать контакт кнопки как вход

}

/*

* Функция сглаживания дребезга

* принимает в качестве аргумента предыдущее состояние кнопки

* и выдает фактическое.

*/

boolean debounce(boolean last)

{

boolean current = digitalRead(BUTTON);// Считать состояние кнопки

if (last != current)// Если изменилось...

{

delay(5);// Ждем 5 мс

current = digitalRead(BUTTON);// Считываем состояние кнопки

return current;// Возвращаем состояние кнопки

}

}

- 57 -

void loop()

{

currentButton = debounce(lastButton);

if (lastButton == LOW && currentButton == HIGH) // Если нажатие

{

ledOn = !ledOn;// Инвертировать значение состояния светодиода

}

lastButton = currentButton;

digitalWrite(LED, ledOn);// Изменить статус состояния светодиода

}

Теперь рассмотрим текст листинга 2.5 подробнее. Сначала заданы номера контактов для подключения кнопки и светодиода. Затем объявлены три глобальные логические переменные, которые будут изменяться в программе (значение глобальной переменной можно менять в любой части программы). Каждой из трех переменных присвоены начальные значения (LOW, LOW и false). Далее в программе значения этих переменных могут изменяться с помощью оператора присваивания =.

Рассмотрим функцию подавления дребезга кнопки boolean debounce(). Эта функция принимает логическую переменную (имеющую только два состояния: true/false, HIGH/LOW, вкл./выкл., 1/0) предыдущего состояния кнопки и возвращает текущее значение состояния кнопки. Внутри функции текущее состояние кнопки сравнивается с предыдущим с помощью оператора != (не равно). Если состояния отличаются, то кнопка, возможно, нажата. Затем ожидаем 5 мс (этого достаточно, чтобы состояние кнопки стабилизировалось после дребезга), прежде чем проверить состояние кнопки снова. Затем вновь проверяем состояние кнопки. Как вы помните, функции могут возвращать результат. Данная функция возвращает текущее значение булевой локальной переменной, которая объявлена и используется только в функции debounce(). Когда функция debounce() вызывается из основного цикла, возвращенное значение записывается в глобальную переменную currentButton, которая была определена в начале программы.

После вызова функции debounce() и установки значения переменной currentButton происходит сравнение текущего и предыдущего значений состояния кнопки с помощью оператора && (логический оператор "И", означающий, что выражение в скобках выполнится, только если истинно каждое из равенств, разделенных оператором &&).

Если ранее состояние кнопки было LOW, а теперь HIGH, значит, кнопка была нажата и нужно инвертировать значение переменной lecton. Это действие выполняет опера-

- 58 -

тор перед переменной lecton. Цикл закончен, обновляем предыдущую перемеую состояния кнопки и изменяем состояние светодиода.

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

2.9. Создание управляемого ночника на RGB-светодиоде

Вы уже знаете, как управлять цифровыми выходами, как создать противодребезговую защиту для кнопки, как менять яркость светодиода с помощью ШИМ-сигнала.