Инкрементальный энкодер (rotary encoder) – это тип электромеханического датчика, который можно использовать для определения углового положения его оси. Данный датчик формирует на своем выходе электрические сигналы в зависимости от направления вращения своей оси. Энкодер состоит из механических компонентов, поэтому он отличается надежностью работы и используется в различных приложениях: роботы, компьютерные мышки, плоттеры, принтеры и т.д.
Существует два типа энкодеров: абсолютные энкодеры и инкрементальные энкодеры. В данной статье мы будем использовать инкрементальный энкодер.
Ранее на нашем сайте мы уже рассматривали подключение инкрементального энкодера к плате Arduino, но эта статья более свежая и значительно более подробная. Также на нашем сайте есть статьи про подключение инкрементального энкодера к другим микроконтроллерам:
Необходимые компоненты
- Плата Arduino Uno (купить на AliExpress).
- Инкрементальный энкодер (купить на AliExpress).
Реклама: ООО "АЛИБАБА.КОМ (РУ)" ИНН: 7703380158
Принципы работы инкрементального энкодера
Распиновка инкрементального энкодера
Модуль инкрементального энкодера содержит 5 контактов: GND, +(VCC), SW(Switch), DT и CLK. Все эти контакты являются цифровыми, за исключением контактов VCC и Ground. Назначение контактов (распиновка) инкрементального энкодера представлена на следующем рисунке.
GND – контакт общего провода (земли) инкрементального энкодера, в нашем проекте его необходимо подключить к общему проводу (ground pin) платы Arduino.
VCC – контакт для подачи питания на инкрементальный энкодер, его необходимо подключать к источнику напряжения питания 5V или 3.3V.
SW (switch pin) – контакт переключения энкодера. Инкрементальный энкодер, который мы используем, содержит внутри себя кнопку, которая активируется при нажатии на верхний край его рукоятки (оси). В активном состоянии эта кнопка формирует уровень LOW.
DT – данный контакт представляет собой фактически то же самое что и контакт CLK, но последовательность импульсов на нем сдвинута по фазе на 90 градусов относительно контакта CLK.
CLK – главный контакт модуля инкрементального энкодера. Каждый раз при вращении рукоятки выходной импульс на нем сменяет свое состояние с High на Low.
Как работает инкрементальный энкодер
Если посмотрим внутрь инкрементального энкодера, то там мы увидим диск с большим количеством вырезанных в нем отверстий. Логика выходных импульсов энкодера определяется его двумя контактами, показанными на рисунке ниже. Пластина энкодера подключается к общему проводу схемы.
При вращении оси энкодера его контакты Output A и Output B вступают в контакт с его пластиной в определенном порядке, который зависит от направления вращения оси энкодера. Если ось энкодера вращается по часовой стрелке, то сначала с пластиной соединяется output A, а потом output B. Если ось энкодера вращается против часовой стрелки, то все происходит с точностью наоборот. Таким образом, при любом направлении вращения оси энкодера сигналы на его выходах output A и output B сдвинуты на до градусов относительно друг друга.
На представленной анимации если мы вращаем ось энкодера по часовой стрелке, контакт A подключается первым, а потом подключается контакт B. Отслеживая какой контакт подключился к пластине первым мы можем определить направление вращения оси энкодера и, соответственно, мы можем инкрементировать или декрементировать состояние счетчика в нашей программе.
Компоненты модуля инкрементального энкодера
Это очень дешевый и простой в использовании модуль, который может быть использован в разнообразных приложениях. В зависимости от направления вращения своей оси он генерирует последовательность цифровых сигналов на своих контактах DATA и clock. Если импульсы начинаются с контакта data, а импульсы на контакте CLK сдвинуты относительно него на 90 градусов, это значит что ось энкодера вращается по часовой стрелке, иначе она вращается против часовой стрелки.
Если вы посмотрите внимательно на модуль инкрементального энкодера, то вы увидите, что на его печатной плате сравнительно мало элементов: на фронтальной части мы видим рукоятку энкодера и контакты, а на тыльной части только подтягивающие резисторы. Фактически, инкрементальный энкодер – это механическое устройство, способное работать с уровнями напряжения 3.3V и 5V.
Наиболее часто задаваемые вопросы про инкрементальный энкодер
Где применяются инкрементальные энкодеры?
Достаточно часто они применяются для управления скоростью и направлением движения ленты транспортёра. Также они востребованы в системах распределения товаров на складах, системах обработки багажа и упаковочных системах.
Какие типы энкодеров существуют?
Энкодеры выпускаются 4-х основных типов: механические, оптические, магнитные и электромагнитные.
Какие сигналы на выходе энкодера?
Инкрементальные энкодеры, также называемые энкодерами с ручкой, представляют собой электромеханические устройства, которые преобразуют угловое положение своей оси в аналоговые или цифровые выходные сигналы. Существует два основных типа подобных энкодеров: абсолютные и инкрементальные.
Что лучше: инкрементальный энкодер или потенциометр?
Основным преимуществом инкрементальных энкодеров перед потенциометрами является то, что они могут вращаться в одном направлении бесконечно долго, что позволяет значительно увеличить точность регулировки необходимого параметра. А потенциометры могут совершать в обоих направлениях только по одному обороту, что не всегда удобно для управления какими либо процессами.
Что такое джиттер энкодера?
Джиттер (дрожание) сигнала энкодера заключается в непостоянстве формы импульсов на выходах двух каналов энкодера относительно друг друга, что приводит к нежелательному смещению импульса вперед или назад.
Схема модуля инкрементального энкодера
Модуль инкрементального энкодера состоит из легко доступных компонентов общего назначения, в простейшем случае это сам энкодер и три подтягивающих резистора. Схема энкодера показана на следующем рисунке.
Схема проекта
Схема подключения инкрементального энкодера к плате Arduino представлена на следующем рисунке.
Как видите, схема подключения достаточно проста, необходимо всего лишь подключить контакты CLK и DT инкрементального энкодера к контактам внешних прерываний D2 и D3 платы Arduino. Также необходимо подключить контакт SW энкодера к контакту D4 платы Arduino – этот контакт мы также будем конфигурировать для обработки внешнего прерывания.
Внешний вид собранной конструкции проекта представлен на следующем рисунке.
Объяснение программы для Arduino
Полный код программы приведен в конце статьи, здесь же мы кратко рассмотрим его основные фрагменты.
В коде нашей программы для обработки угловых координат оси энкодера мы будем использовать прерывания, поскольку в этом случае они значительно уменьшают вероятность ошибки.
Первым делом в программе мы подключим необходимые библиотеки и назначим контактам платы Arduino, к которым подключены контакты энкодера, осмысленные имена.
1 2 3 4 |
#include "PinChangeInterrupt.h" #define CLK 2 #define DT 3 #define SW 4 |
Далее объявим все необходимые переменные.
1 2 3 4 |
int counter = 0; int currentState; int initState; unsigned long bebounceDelay = 0; |
Затем, в функции void setup(), мы зададим режимы работы используемых контактов (на ввод данных, и на ввод данных с использованием внутреннего подтягивающего резистора для контакта SW) и инициализируем последовательную связь для целей отладки.
1 2 3 4 5 6 |
void setup() { pinMode(CLK, INPUT); pinMode(DT, INPUT); pinMode(SW, INPUT_PULLUP); // Setup Serial Monitor Serial.begin(9600); |
После этого мы будем считывать текущее состояние контакта CLK энкодера и сохранять его в переменной initState. Это весьма важный шаг при работе с энкодером и без него счетчик инкрементального энкодера работать не будет.
1 2 |
// Read the initial state of CLK initState = digitalRead(CLK); |
Наконец, в функции setup мы настроим обработку прерываний на используемых нами контактах. Прерывания у нас будут срабатывать при любом изменении состояния контакта (CHANGE), поэтому мы будем схватывать все прерывания от инкрементального энкодера. Более подробно о работе с прерываниями в платах Arduino вы можете прочитать в этой статье.
1 2 3 4 |
attachInterrupt(0, encoder_value, CHANGE); attachInterrupt(1, encoder_value, CHANGE); attachPCINT(digitalPinToPCINT(SW), button_press, CHANGE); } |
Функция void loop() будет у нас пустой поскольку в ней мы не будем производить никаких действий.
1 2 |
void loop(){ } |
Также мы запрограммируем функцию button_press(), которая будет автоматически вызываться при срабатывании прерывания. Внутри этой функции мы будем проверять состояние контакта SW. Если на нем уровень low, мы будем еще ждать некоторое время (это необходимо для борьбы с эффектом дребезга контактов) чтобы проверить действительно ли нажата кнопка или нет. Если кнопка нажата, то в окно монитора последовательной связи мы будем передавать сообщение "Button pressed!".
1 2 3 4 5 6 7 8 9 10 11 |
void button_press() { int buttonVal = digitalRead(SW); //If we detect LOW signal, button is pressed if (buttonVal == LOW) { if (millis() - bebounceDelay > 200) { Serial.println("Button pressed!"); } debounceDelay = millis(); } } |
Еще мы запрограммируем функцию encoder_value(). В ней мы будем проверять позицию энкодера и направление его вращения и в соответствии инкрементировать или декрементировать значение счетчика (переменная counter). Если ось энкодера будет вращаться по часовой стрелке мы будем инкрементировать значение счетчика, а если против часовой – то декрементировать.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
void encoder_value() { // Read the current state of CLK currentState = digitalRead(CLK); // If last and current state of CLK are different, then we can be sure that the pulse occurred if (currentState != initState && currentState == 1) { // Encoder is rotating counterclockwise so we decrement the counter if (digitalRead(DT) != currentState) { counter ++; } else { // Encoder is rotating clockwise so we increment the counter counter --; } // print the value in the serial monitor window Serial.print("Counter: "); Serial.println(counter); } // Remember last CLK state for next cycle initState = currentState; } |
Тестирование работы проекта
В представленном ниже видео показан принцип работы инкрементального энкодера. Слева на нем вы видите что мы вращаем ось энкодера по часовой стрелке, а справа – увеличивающееся значение счетчика в окне монитора последовательной связи. Потом мы нажимаем кнопку энкодера и видим в окне монитора последовательной связи соответствующее появившееся сообщение.
Проблемы, которые могут возникнуть при работе с инкрементальным энкодером
- проверьте хорошо ли контакты VCC и общего провода (земли) инкрементального энкодера подключены к соответствующим контактам платы Arduino;
- если вы работаете с энкодером без использования прерываний, то существует вероятность того, что какие-нибудь операции в основном цикле программы loop могут привести к такой задержке, из-за которой плата Arduino не сможет корректно отрабатывать данные от энкодера;
- если на выходе энкодера появляется достаточно много импульсов шума, это свидетельствует о том что энкодер старый и уровень его джиттера достаточно большой. Единственное решение этой проблемы – заменить энкодер;
- поскольку инкрементальной энкодер почти никак не защищен о воздействия внешней среды, то рядом с его осью может набиться достаточно много пыли, что сделает его бесполезным. Решением данной проблемы является чистка энкодера.
Исходный код программы (скетча)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 |
#include "PinChangeInterrupt.h" #define CLK 2 #define DT 3 #define SW 4 int counter = 0; int currentState; int initState; unsigned long debounceDelay = 0; void setup() { pinMode(CLK, INPUT); pinMode(DT, INPUT); pinMode(SW, INPUT_PULLUP); // Setup Serial Monitor Serial.begin(9600); // считываем состояние контакта CLK initState = digitalRead(CLK); // Call encoder_value() when any high/low changed seen // on interrupt 0 (pin 2), or interrupt 1 (pin 3) attachInterrupt(0, encoder_value, CHANGE); attachInterrupt(1, encoder_value, CHANGE); attachPCINT(digitalPinToPCINT(SW), button_press, CHANGE); } void loop() { } void button_press() { int buttonVal = digitalRead(SW); // Если на контакте уровень LOW, значит кнопка нажата if (buttonVal == LOW) { if (millis() - debounceDelay > 200) { Serial.println("Button pressed!"); } debounceDelay = millis(); } } void encoder_value() { // считываем текущее состояние CLK currentState = digitalRead(CLK); // если последнее и текущее состояния CLK отличаются, значит появился импульс (энкодер вращается) if (currentState != initState && currentState == 1) { // энкодер вращается против часовой стрелки, декрементируем счетчик if (digitalRead(DT) != currentState) { counter ++; } else { // энкодер вращается по часовой стрелке, инкрементируем счетчик Encoder is rotating clockwise so we increment the counter counter --; } // выводим значение счетчика в окно монитора последовательной связи Serial.print("Counter: "); Serial.println(counter); } // запоминаем последнее состояние CLK для следующего цикла initState = currentState; } |
Все необходимые файлы для этого проекта вы можете скачать по следующей ссылке.
3 837 просмотров