Professional Documents
Culture Documents
МПЗА лабораторні АРДУІНО
МПЗА лабораторні АРДУІНО
Протокол № від р.
ПЕРЕДМОВА.
Методичні вказівки.
3
машинному коді), реалізований у вигляді однієї мікросхеми або комплекту з
декількох спеціалізованих мікросхем.
Що таке Arduino?
Апаратна частина.
♦ кнопку скидання;
4
Рис1.1. Компоненти плато Arduino Uno.
5
виробнику, і дозволяє завантажувати призначену для користувача програму
на плату Arduino по послідовному порту USART. У разі Arduino Uno
інтерфейсом між кабелем USB і контактами USART на основному
мікроконтролері служить додатковий контролер ATmega 16U2.
Джерела живлення.
пристроїв.
6
Підключить Arduino до комп’ютеру за допомогою кабелю USB. Тепер
запустіть Arduino IDE. Все готово для завантаження першої програми на
плату Arduino. Щоб переконатися в цьому, запустимо програму Blink, яка
буде блимати вбудованим світлодіодом. На більшості плат Arduino є
світлодіод, підключений до цифрового контакту 13. Виконуємо послідовність
команд File ->Examples -> Basic і вибираємо програму Blink. Відкриється
нове вікно з кодом цієї програми. Завантажимо її в плату Arduino як приклад,
а потім проаналізуємо, щоб зрозуміти, як писати власні програми. Перш ніж
завантажувати програму в плату Arduino, необхідно вказати тип плати і
номер послідовного порту. Знаходимо в меню опцію
ПОРАДА.
7
Структура програми для Arduino.
setup ()
loop ()
Після виклику функції setup () - управління переходить до функції loop (), яка
робить в точності те, що означає її ім'я - безперервно виконується,
дозволяючи програмі щось змінювати, відповідати і управляти платою
Arduino.
Детально розглянемо текст програми Blink (рис. 1.2), щоб зрозуміти базову
8
структуру програм, написаних для Arduino.
♦ 4 - функція setup (), одна з двох функцій, які повинні бути включені в
кожну програму Arduino. Функція - це фрагмент коду, що виконує певне
завдання. Код в тілі функції setup () виконується один раз на початку
програми. Це корисно для установки початкових параметрів настроювання,
призначення режимів портів введення-виведення, ініціалізації
комунікаційних інтерфейсів і т. д.
Зауваження.
Якщо пін сконфігуровано функцією pinMode () в режимі OUTPUT, його
напруга буде встановлено на відповідне значення: 5V (або 3.3V при
під’єднані з піном 3.3V платі) для HIGH та 0V (земля) для LOW.
10
Якщо ж пін налаштований на режим вводу (тобто INPUT), функція
digitalWrite () також дозволить працювати з (HIGH) та (LOW). Але, якщо ви
встановите pinMode () для введення (режим INPUT), і підключити світлодіод
до відповідного піну, при виклику digitalWrite (…,HIGH), світлодіод буде
виглядати тьмяним! Без явного завдання pinMode () в режимі OUTPUT,
digitalWrite () включає внутрішній підтягуючий резистор, який діє як великий
токообмежуючий резистор.
Завдання до роботи.
1. Що таке Arduino?
2. Який процесор використаний на платі? Якої фірми? Яка його тактова
частота?
3. Яку плату Arduino ми використовуємо? Які плати існують іще?
4. Як підключається Arduino до комп’ютеру?
5. Як «заливається» програма до контролеру?
6. Яка структура програми Arduino?
7. Що таке пін?
8. З якими функціями Arduino мала справу перша програма?
9. Модифікуйте програму так, щоб світлодіод світився подаючи ваше
прізвище азбукою Морзе.
10. Навіщо потрібна функція setup ()?
11.Навіщо потрібна функція loop ()?
12.Що таке команда pinMode ()? Які в неї параметри?
11
13.Що таке команда digitalWrite ()? Які в неї параметри?
14.Навіщо потрібна функція delay ()?Які в неї параметри?
12
Рис 2.1. Електричні з’єднання макетної плати.
13
Рис.2.2. Підключення світлодіоду.
♦ струм тече від точки з більш високою потенційною енергією, щоб знизити
потенційну енергію. Користуючись аналогією, електричний струм можна
представити, як потік води, а напруга - як висоту перепаду. Вода (або струм)
завжди тече з точки з більшою висотою (більш висока напруга) до точки з
меншою висотою (або більш низьку напругу). Струм, як вода в річці, завжди
буде йти по шляху найменшого опору в ланцюзі;
♦ за аналогією опір є отвором для протікання струму. Коли вода (струм) тече
через вузьку трубу, за однакову кількість часу проходить меншу кількість,
ніж через широку трубу. Вузька труба еквівалентна більшого опору, тому що
вода буде текти повільніше. Широка труба еквівалентна малому опору, тому
що вода (струм) може текти швидше. Закон Ома визначається наступним
чином:
14
U=I*R
Р = U*I
15
Для резистора зі схеми на рис. 2.3 при падінні напруги 3 В і силі струму
15мА потужність дорівнює:
Лістинг 2.1.
Використання циклу.
17
Звертаємо увагу на те, що після кожного оператору програми стоїть крапка з
комою - символ «;». Оператор циклу for завжди містить три вирази,
розділені крапкою з комою:
♦ перший вираз привласнює початкове значення змінної-лічильника циклу. У
нашому прикладі змінна i отримує початкове значення 100;
♦ другий вираз вказує умову, коли цикл повинен зупинитися. Оператори в
тілі циклу будуть виконуватися знову і знову, поки умова істинна. Запис
«<=» означає «менше або дорівнює». Таким чином, цей цикл буде
виконуватися доти, поки змінна i менше або дорівнює 1000;
19
Що буде відбуватися з світлодіодом при виконанні лістингу 2.3? Ви будете
спостерігати, як світіння світлодіода змінюється від тьмяного до яскравого в
одному циклі for, а потім від яскравого до тьмяного в іншому циклі for. Все
це буде відбуватися в основному циклі loop () до нескінченності. Обов'язково
зверніть увагу на відмінність двох циклів for. У першому циклі вираз i ++ є
скороченням коду i = i + 1. Аналогічно, запис i-- еквівалентна коду i = i-1.
Перший цикл плавно запалює світлодіод до його максимальної яскравості,
другий - поступово гасить його.
20
♦ значення 255 - скважність 100% (завжди HIGH);
♦ значення 127 відповідає скважності 50% (половина часу HIGH, половина
часу LOW).
На графіках рис. 2.4 видно, що для сигналу з скважністю 25% значення HIGH
діє протягом чверті періоду, а решта 75% часу встановлено значення LOW.
Частота прямокутної послідовності імпульсів в разі з Arduino становить
приблизно 490 Гц. Іншими словами, рівень сигналу змінюється від високого
(5 В) до низького (0 В) приблизно 490 раз кожну секунду. Як бачимо,
напруга, що подається на світлодіод, насправді не знижується. Чому ж при
зменшенні скважності спостерігається спад яскравості світіння світлодіоду?
Це пов'язано з особливістю нашого зору. Якщо світлодіод включається і
вимикається один раз за 1 мс (при скважності 50%), то вам здається, що
яскравість світіння світлодіода становить приблизно 50% від максимальної,
тому що перемикання відбувається швидше, ніж очі можуть це зафіксувати.
Ваш мозок фактично усереднює сигнал і створюється враження, що
світлодіод працює на половині яскравості.
21
Рис.2.5. Підключення світлодіоду та кнопки до плати Arduino.
ПОРАДА.
Рис.2.5.1
22
можуть привести до того, що значення напруги буде коливатися між HIGH і
LOW. Щоб запобігти цьому, стягуючий резистор підключають так, як
показано на рис. 2.5.1.
ПРИМІТКА.
23
void loop ()
{
if (digitalRead (BUTTON) = = LOW)
{
digitalWrite (LED, LOW);
}
else
{
digitalWrite (LED, HIGH);
}
}
Зауваження.
HIGH = true=1
LOW =false=0
24
так легко, як здається. Не можна просто зчитувати значення сигналу на вході,
необхідно враховувати явище, зване брязкотом контактів. Звичайні кнопки
являють собою механічні пристрої з пружинним контактом. При натисканні
на кнопку сигнал не просто змінюється від низького до високого, він
протягом декількох мілісекунд неодноразово змінює своє значення, перш ніж
установиться рівень HIGH. Відмінність очікуваного процесу від реального
ілюструють осцилограми сигналу з кнопки, наведені на рис. 2.6.
25
Основна ідея в тому, що після натискання кнопки треба зачекати деякий час
– не менше ніж 5мс! - перед тим як аналізувати новий стан.
void setup ()
{
pinMode (LED, OUTPUT); // Конфігурувати контакт світлодіода як вихід
pinMode (BUTTON, INPUT); // Конфігурувати контакт кнопки як вхід
}
/*
* Функція згладжування брязкоту
* Приймає в якості аргументу попередній стан кнопки
* І видає фактичне.
*/
void loop ()
{
currentButton = debounce (lastButton);
if (lastButton == LOW && currentButton == HIGH) // Якщо натискання
{
ledOn = !ledOn; // Інвертувати значення стану світлодіода
}
lastButton = currentButton;
digitalWrite (LED, ledOn); // Змінити статус стану світлодіоду
}
27
локальної змінної, яка оголошена і використовується тільки в функції
debounce (). Коли функція debounce () викликається з основного
циклу,повернене значення записується в глобальну змінну currentButton, яку
було визначено на початку програми. Після виклику функції debounce () та
установки значення змінної currentButton відбувається порівняння поточного
і попереднього значень стану кнопки за допомогою оператора && (логічний
оператор "І", що означає, що вираз в дужках виконується, тобто має значення
«істина» тільки якщо істинно кожна з рівностей, розділених оператором &&).
Якщо раніше стан кнопки було LOW, а тепер HIGH, значить, кнопка була
натиснута і потрібно інвертувати значення змінної ledOn. Ця дія
(інвертування) виконує оператором «!» перед змінною ledon (логічна функція
«НІ»). Цикл закінчений, оновлюємо попередню змінну стану кнопки і
змінюємо стан світлодіоду. Програма змінює стан світлодіода після кожного
натискання кнопки. При відсутності перевірки брязкоту кнопки результати
будуть непередбачуваними.
Завдання по роботі.
В кожній з робіт потрібно розробити схеми та заставити її працювати за
правилами, що викладені в задачах1-4. До кожної із задач у зошиті
повинні бути намальовані відповідні схеми! Всюди n – номер варіанту.
Задача №1. «Боротьба» з брязкотом кнопки.
5 , n=0 mod 5
{6 , n=1 mod 5
t= 7 , n=2 mod 5
8 , n=3 mod 5
9 , n=4 mod 5
28
крок – 100/(1 + n mod 5) (округлити до цілого числа);
кінцеве значення параметру циклу 200* (n mod 6 + 1).
Скласти проекти по зміні яскравості світлодіоду від min до max, та від max
до min з кроком 1.
min = n,
max = 255-n.
boolean – логічний
byte - тип даних byte 8-ме бітове беззнакове ціле число, в діапазоні 0..255.
Приклад.
byte c = B10010; // "B" префікс двійкової системи числення (B10010 = 18 в
десятковій системі числення)
int - тип даних int (від англ. Integer - ціле число) один з найбільш часто
використовуваних типів даних для зберігання чисел. int займає 2 байта
пам'яті, і може зберігати числа від -32 768 до 32 767 (від -2 ^ 15 до 2 ^ 15-1)
word - тип даних word зберігає 16-бітове, що не містить знака, число від 0
до 65535. Теж саме, що unsigned int - (беззнаковое ціле число).
Приклад.
word w = 10000;
29
unsigned int
float - тип даних float служить для зберігання чисел з плаваючою комою.
Цей тип часто використовується для операцій з даними, що зчитуються з
аналогових входів. Діапазон значень від -3.4028235E + 38 до 3.4028235E + 38.
Змінна типу float займає 32 біта (4 байта) в пам'яті.
double - тип даних double, на відміну від більшості мов програмування,
має ту ж точність, що і тип float і займає також 4 байта пам'яті.Тип double
підтримується в Arduino для сумісності коду з іншими платформами.
string - масив символів
String - об’єкт класу
void - ключове слово void використовується при оголошенні функцій,
якщо функція не повертає ніякого значення при її виклику (в деяких мовах
програмування такі функції називають процедурами).
Приклад.
// В функціях "setup" робляються деякі дії,
// Але нічого не повертається у зовнішню програму
void setup ()
{
// ...
}
Арифметичні оператори в Arduino.
= (Assignment)
+ (Addition)
- (Subtraction)
* (Multiplication)
/ (Division)
% (Modulo)
= = (Equal to)
! = (Not equal to)
<(Less than)
> (Greater than)
<= (Less than or equal to)
> = (Greater than or equal to)
30
&& (І)
|| (АБО)
! (Заперечення)
++ (Increment)
- - (Decrement)
Приклад.
x ++; // Збільшує значення x на одиницю і повертає старе значення x
++ х; // Збільшує значення x на одиницю і повертає нове значення x
в обох випадках х стає рівним 6, але ++ х буде рівне 6 до того, як його буде
використано, а x ++ стане рівне 6 після того, як його буде використано.
Приклад.
If (логічний вираз)
{дії}
if (логічний вираз 1)
{дії 1}
31
{дії 2}
{дії n}
else
Функції в Arduino.
Існують дві обов'язкові функції в скетчах Arduino setup () і loop (). Інші
функції повинні створюватися за дужками цих функцій. У наступному
прикладі буде створено проста функція множення двох чисел.
Приклад.
Синтаксис функції Arduino
void loop () {
int i = 2;
32
int j = 3;
int k;
k = myMultiplyFunction (i, j); // k містить 6
int myMultiplyFunction(int x, int y){
int result;
result = x * y;
return result;
}
Створену функцію необхідно задекларувати поза дужками будь-який інший
функції, таким чином "myMultiplyFunction ()" може стояти вище або нижче
функції "loop ()".
34
Короткі теоретичні відомості та методичні вказівки по виконанню
роботи.
Дані про навколишній світ всі пристрої неминуче отримують в аналоговому
вигляді. Цифрова інформація являє собою серію бінарних (цифрових)
даних. Кожний біт приймає тільки одне з двох можливих значень. Графіки
на рис. 3.1 показують, чим відрізняються один від одного аналогові і цифрові
сигнали. Зліва прямокутні імпульси, амплітуда яких бере тільки два
значення: 0 і 5 вольт. Точно так само, як з кнопкою з попередньої роботи:
тільки HIGH або LOW. Справа зображений фрагмент синусоїдального
сигналу. Незважаючи на те, що його амплітуда знаходиться в тих же межах
(0 і 5 вольт), аналоговий сигнал приймає нескінченне число значень між
цими двома значеннями 0 та 5.
35
Перетворення аналогового сигналу в цифровий.
Зауваження.
37
Перш ніж використовувати потенціометр для управління іншим
обладнанням, подивимося, як зчитати значення опору потенціометра за
допомогою АЦП і передати через послідовний порт Arduino для перегляду
значень на комп'ютері. Для читання значення аналогового входу передбачена
функція analogRead (), для виведення значень в послідовний порт Arduino
IDE - функція serial.println(). Наберіть і завантажте в плату Arduino програму
з лістингу 3.1.
Коментар до лістингу.
ПРИМІТКА.
39
Якщо виводяться незрозумілі символи, переконайтеся, що швидкість
передачі даних установлена правильно. У програмі порт ініціалізований на
швидкість 9600 бод, таке ж значення необхідно встановити в настройках
монітору послідовного порту.
Позначення на схемі
40
Рис.3.6. Схема підключення датчика температури до аналогового входу 0.
Лістинг 3.2.
int val;
int tempPin = 0;
void setup()
{
Serial.begin(9600);
}
void loop()
{
val = analogRead(tempPin);
float mv = ( val/1023.0)*5000;
float cel = mv/10;
float farh = (cel*9)/5 + 32;
Serial.print("TEMPRATURE = ");
Serial.print(cel);
Serial.print("*C");
Serial.println();
delay(1000);
41
/* uncomment this to get temperature in farenhite
Serial.print("TEMPRATURE = ");
Serial.print(farh);
Serial.print("*F");
Serial.println();
delay(1000);
*/
}
Коментар до лістингу.
Рис.3.8. Фоторезистор.
44
виглядає наступним чином:
Тут:
і 900, відповідно, a toLow і toHigh рівні 255 і 0, то200 перейде у 255, 900 у 0, а
550 перетвориться в 127, тому що 550 знаходиться посередині між 200 і 900,
а 127 посередині між 255 і 0.
Лістинг 3.3.
45
// Автоматичний нічник
const int RLED = 9; // Контакт 9 для ШІМ - виведення RED RGB-світлодіоду
const int LIGHT = 0; // Контакт А0 для входу фоторезистору
const int MIN_LIGHT = 200; // Нижній поріг освітленості
const int MAX_LIGHT = 900; // Верхній поріг освітленості
int val = 0; // Змінна, для збереження зчитаного аналогового значення
void setup ()
{
pinMode (RLED, OUTPUT); // Конфігурувати RED-контакт світлодіоду як
//вихідного
}
void loop ()
{
val = analogRead (LIGHT); // Читання показань фоторезистору
val = map (val, MIN_LIGHT, MAX_LIGHT, 255, 0); // Виклик функції map ()
val = constrain (val, 0, 255); // Обмеження меж
analogWrite (RLED, val); // Управління світлодіодом
Завдання по роботі.
В кожній з робіт потрібно розробити схеми та заставити її працювати за
правилами, що викладені в задачах. До кожної із задач у зошиті повинні
бути намальовані відповідні схеми! Всюди n – номер варіанту. Тексти
програм записати у зошит.
Задача №1.
46
Зберіть схему, що містить в собі фоторезистор та світлодіод, та напишіть
програму для нічного світильника , яка б реагувала на рівень освітленості та
включала/виключала світлодіод коли на аналоговий вхід Arduino подається
напруга:
Включення світлодіоду:
напруга = 20/(∑ (прізвище)(v)
Виключення світлодіоду:
напруга = 40/(∑ (прізвище)(v)
Питання по роботі.
Методичні вказівки.
Що таке сервопривід.
Крутити мотор досить цікаво, але коли ми робимо проект, в якому
необхідний контроль руху, хочеться чогось більшого.
Двигуни постійного струму прекрасно діють в якості моторів, але дуже
незручні для точних робіт, т. к. не мають зворотного зв'язку. Іншими
словами, без якогось зовнішнього датчика можна дізнатися положення вала
двигуна постійного струму. Серводвигуни (або сервоприводи), навпаки,
відрізняються тим, що за допомогою команд можна встановити їх в певне
положення, в якому вони знаходитимуться до надходження нових команд. Це
важливо, коли необхідно деяку систему перемістити в певне положення,
повернувши на деякий кут. У цьому розділі ви дізнаєтеся про серводвигуном
і їх управлінні за допомогою Arduino.
Розглянемо приклад управління сервоприводом. Сервоприводи масово
виробляються, легко доступні, вартість становить від пари доларів до сотень.
Усередині сервоприводу вбудований невеликий редуктор (щоб збільшити
потужність) і електроніка (для спрощення управління). Стандартний
сервопривід позиціонується від 0 до 180 градусів.
48
Позиція задається тривалістю керуючого імпульсу, від 1мс (0 градусів)
до 2 мс (180 градусів), 1.5 мс( для 90 градусів). Тимчасові параметри
тривалості імпульсу можуть відрізнятися у різних виробників. Якщо
посилати імпульси кожні 25 ... 50 мс, то сервопривід може плавно
обертатися. Одним з переваг Arduino є готова до використання бібліотека
підпрограм «Servo.h», що дозволяє легко управляти двома сервоприводами
Лістинг 4.1.
Повороти від 00 до 1800 та від 180° до 0° в циклі зсув по 10.
49
// Поворот сервоприводу (by BARRAGAN <http://barraganstudio.com>)
#include <Servo.h> // підключення до проекту бібліотеки Servo.h.
Servo myservo; // створення об’єкту Servo (“серво”) з ім’ям myservo
// за сервоприводом для контролю
int pos = 0; // змінна для зберігання позиції серво
void setup() {
myservo.attach(9); // призначає вивід 9 як керуючий для об’єкту myservo
}
void loop() {
for(pos = 0; pos < 180; pos += 1)
{
// переміщення від 0° до 1800 з кроком в 1
myservo.write(pos); //команда сервоприводу
//перейти у позицію із змінною 'pos'
delay(15); // затримка 15мс for the servo to reach the position
}
for(pos = 180; pos>=1; pos-=1) // переміщення від 180° до 0°
{
myservo.write(pos); //команда сервоприводу
//перейти в позицію із змінною 'pos'
delay(15); // затримка 15мс для досягнення заданої позиції
}
Лістинг 4.2.
#include <Servo.h>
Servo myservo; // створення об’єкту “серво” для контролю сервоприводу
int pos = 0; // змінна для зберігання позиції серво
void setup() {
myservo.attach(9); // призначає вивід 9 як керуючий для об’єкту серво
}
void loop() {
pos = 0;
delay(1000);
myservo.write(pos); //команда сервоприводу перейти у позицію із змінною 'pos'
delay(1000);
50
pos=30;
myservo.write(pos); //команда сервоприводу
//перейти у позицію із змінною 'pos'
delay(1000);
pos=60;
myservo.write(pos); //команда сервоприводу
//перейти у позицію із змінною 'pos'
delay(1000);
pos=90;
myservo.write(pos); //команда сервоприводу
//перейти у позицію із змінною 'pos'
delay(1000);
}
Кроковий двигун 28BYJ-48 з драйверомULN2003 та Arduino Uno.
Кроковий двигун може точно переміщатися на мінімально можливий кут,
званий кроком. Для практичних завдань можна вважати, що кроковий мотор
трохи схожий на сервопривід. Можна задати йому повернутися в деякий стан
і можна розраховувати отримати досить стабільні результати в декількох
повторних експериментах. Зазвичай, сервоприводи обмежені кутом повороту
в діапазоні від 0 до 180 °, кроковий же двигун може обертатися безперервно,
подібно двигуну постійного струму. Перевагою крокових двигунів є те, що
можна досягти набагато більшому ступені контролю над рухом. До
недоліком крокових двигунів можна віднести кілька більш складне
управління, ніж у випадках з сервоприводами або моторами постійного
струму.
28BYJ-48 - це маленький, дешевий, 5 вольтовий кроковий моторчик з
редуктором. Передавальне число редуктора у нього приблизно 64: 1, що
дозволяє отримати цілком гідний крутний момент для моторчика такого
розміру і швидкість близько 15 об / хв.
КРОК A B A\ B\
0 1 1 0 0
1 0 1 0 0
2 0 1 1 0
3 0 0 1 0
4 0 0 1 1
5 0 0 0 1
6 1 0 0 1
7 1 0 0 0
52
Табл. 4.1. Напівкрокова послідовність сигналів на котушки.
53
Рис. 4.3. Плата керування на базі ULN2003.
Лістинг 4.3.
Лістинг 4.3.
58
// підключити змінний резистор, крутимо резистор - міняємо швидкість
// обертання
// його ножки одна 5v, інша GND, третя А0
Int VariableResistor = analogRead(A0);
motorSpead = map(VariableResistor,0,1023,10000,1200);
Завдання по роботі.
В кожній з робіт потрібно розробити схеми та заставити її працювати за
правилами, що викладені в задачах. До кожної із задач у зошиті повинні
бути намальовані відповідні схеми.
Задача №1.
∑ (прізвище)0.
Задача №3.
Задача №6.
Опис типу.
Тип даних byte – це 8-ми бітове беззнаковое ціле число, в діапазоні 0..255.
Приклади.
Arduino і масиви.
Функція bitRead(x,n).
Опис функції.
Читає відповідний біт числа;
синтаксис
bitRead (х, n);
параметри
х: число, з якого читати
n: який біт читати, починаючи з 0 (крайній правий) біту;
повертає значення біта (0 чи 1).
Наприклад,
bitRead (B011101 1,2) = 0
Опис функції.
Призупинення програми на задану кількість часу в мікросекундах, заданого
як параметр x. Є тисячі мікросекунд в мілісекунди, а мільйон мікросекунд в
секунду.
Питання по роботі.
1. Що таке сервопривід?
2. Яка бібліотека дозволяє керувати сервоприводом?
3. Як підключити бібліотеку, що є серед бібліотек вашого Arduino, до
поточного скетчу?
4. Як заставити серводвигун повернутися на потрібний кут?
5. Що таке кроковий двигун?
6. Принцип роботи крокового двигуна.
61
7. Що означають терміни 8 – керуюча послідовність, 64 передаточне
число редуктора, 512 кроків для здійснення повного оберту валу при
роботі з кроковим двигуном?
8. Що таке ULN2003 і як працювати з нею?
9. На який мінімальний кут можна повернути кроковий двигун?
10.Як заставити кроковий двигун повертатися в ту чи іншу сторону?
11. Що таке масив?
12.Як задаються масиви?
13.Як звернутися до потрібного елементу масиву?
14. Що таке байтова змінна?
15.Що таке бінарний код числа? Як його задавати?
16. Як звернутися до потрібного розряду числа, що має запис у двійковій
системі числення?
Мета: навчити роботі з Alphanumeric I2C Liquid Crystal Display (1602 LCD),
за допомогою мікроконтролера Arduino.
Методичні вказівки.
63
Рис.5.1. Схема підключення I2C - пристроїв .
Взаємодія і ідентифікація пристроїв.
Протокол I2C дозволяє декільком веденим пристроям з'єднуватися по одній
шині з одним провідним пристроєм. Далі роль ведучого пристрою (майстра)
буде грати плата Arduino. Майстер шини відповідає за ініціювання обміну.
Підпорядковані пристрої не можуть ініціювати обмін даних, вони здатні
лише відповідати на запити, які посилає ведучий пристрій. Так як до однієї
лінії підключено кілька ведених пристроїв, принципово важливо, що
встановлювати зв'язок може тільки ведучий пристрій. В іншому випадку,
відразу кілька пристроїв могли одночасно намагатися відправити дані, що
призвело б до непорозуміння. Всі команди і запити, надіслані від майстра,
приймаються всіма пристроями на шині. Кожний відомий пристрій має
унікальну 7-бітову адресу (ідентифікаційний номер, ID пристрою). Коли
ведучий пристрій ініціює зв'язок, воно передає ідентифікатор веденого.
Ведені пристрої реагують на дані, що передаються по шині, тільки тоді, коли
вони спрямовані на їхню адресу. Адреса кожного пристрою на шині I2C
повинна бути унікальною. Деякі пристрої I2C мають адреси, що
настроюються, а інші - фіксовані адреси, що задаються виробником. При
підключенні до однієї шині декількох I2C - пристроїв, необхідно щоб у них
були різні! ідентифікатори.
Організація обміну даними з пристроєм по I2C залежить від вимог
конкретного пристрою. На щастя, наявність в Arduino бібліотеки I2C звільняє
від необхідності програмування операцій синхронізації процесу обміну. Далі
ми розглянемо організацію обміну даними з I2C Liquid Crystal Display - LCD -
64
дисплеєм. Цей приклад дозволить в майбутньому легко здійснити обмін з
іншими пристроями I2C.
Лінії інтерфейсу SDA і SCL у МК Atmega8 / 168/328, сидять на ніжках з
номерами 27 (PC4) і 28 (PC5), відповідно. На платах Arduino, лінія даних -
SDA (лінія даних) виведена на аналоговий пін 4, а лінія тактування - SCL
(лінія синхронізації) виведена на аналоговий пін 5. Для роботи з протоколом
I2C, в Arduino є штатна бібліотека Wire, яка дозволяє взаємодіяти з I2C –
пристроями. В той же час однієї цієї бібліотекі недостатньо, щоб працювати з
LCD - дисплеєм. Нам буде потрібна ще одна бібліотека – Liquid Crystal I2C –
якої немає серед набору стандартних бібліотек Arduino.
Як підключити нову бібліотеку до Arduino.
1. Зайти потрібну бібліотеку в Internet.
2. Скачати відповідний архівний «.zip» файл.
3. Знайти в папці Arduino, де розташовані всі папки, що утворені
після інсталяції Arduino, папку «libraries» та розмістити в ній папку,
що утворилася після розархівації відповідної бібліотеки.
4. Нова бібліотека з’явиться автоматично серед бібліотек вашого
Arduino.
Для того, щоб спілкуватися з відповідним пристроєм по I2C обов’язково
потрібно знати його адресу. Цікаво те, що в інструкціях по роботі з LCD –
дисплеєм(1602 LCD) у Arduino вказується адреса 0x27 (0x27 – число 27, що
задано системі числення з основою 16). При перевірці відповідної адреси
пристрою адреса була зовсім іншою, а саме 0x3F. Зрозуміло, що вказавши
неправильну адресу, ми не змогли б працювати з нашим пристроєм.
Як знайти адресу пристрою.
1) Потрібно правильно підключити LCD – дисплей до Arduino, згідно
наступної схеми:
#include <Wire.h>
#include <LiquidCrystal_I2C.h>
LiquidCrystal_I2C lcd(0x3F,16,2); // set the LCD address to 0x3F for a 16
//chars and 2 line display
void setup()
{
lcd.init(); // initialize the lcd
// Print a message to the LCD.
lcd.backlight();
lcd.print("Hello, world!");
}
void loop()
{
}
Дамо деякі пояснення до скетчу.
Перші два рядки - це підключення відповідних бібліотек.
Третій рядок вказує, що працюємо з об’єктом класу LiquidCrystal_I2C, який
буде у нас мати ім’я lcd. Після імені lcd у круглих дужках стоїть:
1) Адреса, що нами визначена для дисплею;
67
2) Кількість знаків (стовпчиків), що можна використовувати на кожному
рядку у цьому дисплеї;
3) Кількість рядків, що можна використовувати у цьому дисплеї.
B00000, };
68
означає, що при роботі з об’єктом lcd, буде створено символ користувача
під номером 2, для символу «+», який задається масивом fan_off.
#include <Wire.h>
#include <LiquidCrystal_I2C.h>
LiquidCrystal_I2C lcd(0x3F,16,2); // set the LCD address to 0x3F for a 16
//chars and 2 line display
// символ користувача «+»
byte fan_off[8] = { B00100, B00100, B00100, B11111, B00100, B00100, B00100,
B00000, };
// символ користувача «*»
byte fan_on[8] = {B00100, B10101, B01110, B11111, B01110, B10101, B00100,
B00000, };
void setup()
{
lcd.init(); // initialize the lcd
// Print a message to the LCD.
lcd.backlight();
lcd.print("Hello, world!");
// визначаємо символи користувача
lcd.createChar(2, fan_off); // число 2 для «+»
lcd.createChar(0, fan_on); //число «0» для «*»
lcd.setCursor(0,1) ;
lcd.write(0) ;
delay(1000);
lcd.setCursor(15,1) ;
lcd.write(2) ;
delay(1000);
}
void loop()
69
{
}
Завдання по роботі.
В кожній з робіт потрібно розробити схеми та заставити її працювати за
правилами, що викладені в задачах. До кожної із задач у зошиті повинні
бути намальовані відповідні схеми.
Задача №1.
//Робота з курсором
lcd.setCursor(0, 0); // Встановлюємо курсор (номер стовчика(позиції), рядок)
lcd.home(); // Установка курсора в нуль (0, 0)
lcd.cursor(); // Увімкнути видимість курсора (підкреслення)
lcd.noCursor(); // Прибрати видимість курсора (підкреслення)
70
lcd.blink(); // Увімкнути моргання курсора (курсор 5х8)
lcd.noBlink(); // Вимкнути моргання курсора (курсор 5х8)
//Виведення інформації
lcd.print("mikrotik.kiev.ua"); // Виведення інформації
lcd.clear(); // Очистка дисплея, (видалення всіх даних) установка
//курсора в нуль
lcd.rightToLeft(); // Запис відбувається справа наліво
lcd.leftToRight(); // Запис відбувається зліва направо
lcd.scrollDisplayRight(); // Зміщення всього зображення на дисплеї на
один //символ вправо
lcd.scrollDisplayLeft(); // Зміщення всього зображення на дисплеї на
один //символ вліво
//Підсвічування
lcd.backlight(); // Вмикання підсвічування
lcd.noBacklight(); // Вимикання підсвічування
Питання по роботі.
Методичні вказівки.
72
Рис. 6.2. Вигляд відповідності контактів і сегментів індикатору.
void setup()
{
for(int i = 2; i <= 8; i++)
{
pinMode(i, OUTPUT);
}
pinMode(BUTTON, INPUT);
}
void loop()
{
currentButton = debounce (lastButton);
if (lastButton == LOW && currentButton == HIGH) // Якщо натискання
{
counter = (counter + 1) % 10;
}
lastButton = currentButton;
writeNumber(counter);
}
75
5. Змінна mask «витягує» черговий елемент масиву numbers[], з
номером number.
6. Змінна currentPinMask має в своєму двійковому записі рівно одну
одиницю.
7. Якщо при побітовій кон’юнкції (mask & currentPinMask) якійсь із
розрядів рівний 1 і в змінній mask і в змінній currentPinMask,
наприклад, розряд 3, то результат операції (mask & currentPinMask)
не дорівнює нулю і ми «запалюємо» сегмент з номером 3. Якщо ж
результат операції (mask & currentPinMask) рівний 0, то ми
виключаємо сегмент 3 нашого індикатору.
8. Операція currentPinMask >> 1 пересуває «1» зліва направо по на
один розряд і ми будемо слідкувати за черговим розрядом
вибраного елементу масиву numbers[].
9. Зауважимо, що відповідні дії робляться в циклі, що крок за кроком
перебирає всі розряди чергового двійкового числа.
сегмент ніжка
A 11
B 7
C 4
D 2
E 1
F 10
G 5
DP 3
За 1 6
розряд
За 2 8
розряд
За 3 9
розряд
За 4 12
розряд
Рис.6.7. Відповідність сегментів і «ніжок» індикатору.
77
функція дозволяє без проблем виводити цілі, дробові і від’ємні числа у
чотирьохрозрядний 7 – сегментний індикатор.
{
if(int(num * 10) != int(num)*10)
78
{
floatingPoint++;
num *= 10;
}
else
break;
}
for(int i = 0, temp; millis() - ltime <= time * 1000; i++)
{
if(i == 4) i = 0;
temp = int(num / pow(10, i)) % 10; // Цифра, що передаємо індикатору
if(num >= 10000 || num <= -10000 || minus == i)
// якщо мінус або переповнення, передаємо '-'
temp = 11;
// працюємо з 4 розрядом
if(i == 3 && (num >= 1000 || floatingPoint == i || minus == i))
pinMode(pin[11], OUTPUT);
else pinMode(pin[11], INPUT);
// працюємо з 3 розрядом
if(i == 2 && (num >= 100 || floatingPoint == i || minus == i))
pinMode(pin[8], OUTPUT); else pinMode(pin[8], INPUT);
// працюємо з 2 розрядом
if(i == 1 && (num >= 10 || floatingPoint == i || minus == i)) pinMode(pin[7],
OUTPUT);
else pinMode(pin[7], INPUT);
// працюємо з 1 розрядом
if(i == 0) pinMode(pin[5], OUTPUT); else pinMode(pin[5], INPUT);
79
for(int j = 0; j < 8; j++) digitalWrite(settingsSegments[j], LOW);
// вимикаємо усі світлодіоди
}
}
void setup()
{
}
void loop()
{
ledDigitDisplay(3.14, 2);
ledDigitDisplay(123, 2);
ledDigitDisplay(-5, 2);
}
Завдання по роботі.
В кожній з робіт потрібно розробити схеми та заставити її працювати
за правилами, що викладені в задачах. До кожної із задач у зошиті
повинні бути намальовані відповідні схеми.
1) Скласти схему та завантажити лістинг 6.1. Впевнитись, що
однорозрядний індикатор працює правильно. Розібратися з кожним
рядком коду!
2) Написати коментар до лістингу 6.1.
3) Скласти схему та завантажити лістинг 6.2. Впевнитись, що
чотирьохрозрядний індикатор працює правильно. Розібратися з
кожним рядком коду!
4) Написати коментар до лістингу 6.2.
5) Розробити скетч з однорозрядним індикатором так, щоб при
натисканні на кнопку крок за кроком виводилися цифри вашої дати
народження. Наприклад, якщо дата «10.11.1998», то виводиться «1»,
«0», «.» і т.д.
80
6) *Розробити скетч з чотирьохрозрядним індикатором так, щоб справа
наліво пересувалася б в нескінченному циклі дата вашого дня
народження. Наприклад, число «10» у попередньому завданні.
Питання по роботі.
Методичні вказівки.
- Arduino
- IR приймач (в нашому випадку VS1838B)(Рис.7.1)
- Перемички і макетна плата для зручності монтажу.
81
Рис.7.1 IR приймач VS1838B та дистанційний пульт.
/*
* IRremote: IRrecvDemo - demonstrates receiving IR codes with IRrecv
* An IR detector/demodulator must be connected to the input RECV_PIN.
* Version 0.1 July, 2009
82
* Copyright 2009 Ken Shirriff
* http://arcfn.com
*/
void setup()
{
Serial.begin(9600);
irrecv.enableIRIn(); // запускаємо приймач до роботи
}
void loop() {
if (irrecv.decode(&results)) { // якщо результат декодування ненульовий
Serial.println(results.value, HEX); // виводимо результат декодування у
шістнадцятковому коді
irrecv.resume();// «скидаємо» приймач, тобто переводимо приймач у стан
готовності прийняти новий сигнал
}
}
0 - FF6897
1 – FF30CF
2 – FF18E7
3 – FF7A85
4 – FF10EF
5 – FF38C7
6 – FF5AA5
7 – FF42BD
83
Звертаємо увагу, що при тестуванні ваших передавачів, коди можуть бути
зовсім іншими.
void setup ()
{
irrecv.enableIRIn (); // включити приймач
pinMode (13, OUTPUT);
pinMode (2, OUTPUT);
pinMode (3, OUTPUT); // призначаємо піни як виходи
}
void loop ()
{
if (irrecv.decode (& results))
{
delay (300); // затримка перед виконанням визначення кнопок, щоб уникнути
швидкого подвійного натискання
if (results.value == 0xff6897) {a = a + 1;} // якщо натиснули кнопку «0» (ff6897
84
код кнопки 0), то змінна «а» збільшується на 1
if (results.value == 0xff30cf) {b = b + 1;} // ВАЖЛИВО !!! «ff30cf» це код
кнопки №1 - мого пульта
if (results.value == 0xff18e7) {c = c + 1;} // ff18e7 це код кнопки №2
irrecv.resume (); //
Завдання по роботі.
світлодіодом.
кроковим двигуном.
LCD – дисплеєм.
85
Задача №3. Робота з власним ІЧ передавачем.
Питання по роботі.
Література.
87