Ви не увійшли.
Доброго дня, Шановні. Я ну прям дуже початківець, допоможіть будь ласка розібратися.
Задача в мене така: одна ардуінка працює як пульт, зчитує джойстик; друга керує двома сервоприводами; зв'язок між ними по UART.
Один байт передавати навчився, але мені потрібно як мінімум 2 (потім більше). Підкажіть, як коректно передавати та приймати масив, зайшов в тупік, приймач не компілюється ні в яких варіантах...
передавач, скомпілювався, але чи вірно Serial.write(buff[2]);
#include <SoftwareSerial.h>
int valud;
int vallr;
int buff[2];
int potpinud=0;
int potpinlr=1;
void setup()
{
Serial.begin(9600,SERIAL_8N1);
pinMode(potpinud,INPUT);
pinMode(potpinlr,INPUT);
}
void loop()
{
valud=analogRead(potpinud);
vallr=analogRead(potpinlr);
valud=map(valud,0,1023,1,180);
vallr=map(vallr,0,1023,1,180);
buff[0]=valud;
buff[1]=vallr;
Serial.write(buff[2]);
delay(20);
}
приймач:
#include <SoftwareSerial.h>
#include <Servo.h>
Servo servoud;
Servo servolr;
int valud;
int vallr;
int buff[2]={0,0};
void setup()
{
Serial.begin(9600,SERIAL_8N1);
servoud.attach(6);
servolr.attach(5);
}
void loop(){
buff=Serial.readBytes();
valud=buff[0];
vallr=buff[1];
if (valud>0) {
valud=valud-1;
servoud.write(valud);}
if (vallr>0) {
vallr=vallr-1;
servolr.write(vallr);}
delay(20);
}
Ніяк не компілює buff=Serial.readBytes();
Остання редакція wsx (2024-03-31 11:10:51)
Неактивний
https://www.arduino.cc/reference/en/lan … readbytes/:
Serial.readBytes(buffer, length)Активний
https://www.arduino.cc/reference/en/lan … readbytes/:
Serial.readBytes(buffer, length)
Одразу так і робив, але не компілюється і все...
Така помилка: Compilation error: no matching function for call to 'HardwareSerial::readBytes(int [2], int)'
Остання редакція wsx (2024-03-31 11:09:59)
Неактивний
А це:
Serial.write(buff[2]);спроба доступа за межі масива buff. Це невизначена поведінка.
Почитайте основи про масиви в C (у C++ також використовуються C-шні масиви).
По книжці Serial.write(buf, len) , але так не компілюється
Serial.write(buff[2]); так пройшло, але мабуть це не вірно...
Читав вже перечитав, мозок пухне, не хоче компілювати... Тому і звернувся по допомогу.
Остання редакція wsx (2024-03-31 11:14:29)
Неактивний
Одразу так і робив, але не компілюється і все...
Така помилка: Compilation error: no matching function for call to 'HardwareSerial::readBytes(int [2], int)'
естественно не компилируется...
функция ожидает размерность Byte 8бит , а вы ей буфер размерностью 16бит подсовываете еще и знаковый.
buffer: the buffer to store the bytes in. Allowed data types: array of char or byte.
Щось я заплутався.. Отже масив з бітів, а не з байтів? який тоді від нього сенс? Один байт можно і так відправити - отримати
Serial.write(val);
val=Serial.read();
Так все працює добре на один серво. Але мені потрібно два байти відправити (а потім ще більше). Як це зробити?
По одному ж не можна відправляти, приймач не зрозуміє хто є хто..
Ну ось же, байт.. або я щось недогоняю?
Serial.write(val), Serial.write(buf, len)
Отправляет в порт val численное значение или строку, или отправляет количество len байт из буфера buf.
Serial.readBytes(buffer, length)
Читает данные из порта и закидывает их в буфер buffer . Также указывается количество байт, который нужно записать – length (чтобы не переполнить буфер).
Остання редакція wsx (2024-03-31 12:11:55)
Неактивний
. Як це зробити?
По одному ж не можна відправляти, приймач не зрозуміє хто є хто..
Свои переменные преобразовать в строку,
Передать: 0х55,0хАА (признак начала посылки) , "строка первой переменой", "строка второй переменой, 0х0D(символ "возврат коретки"окончание посылки)
На приемнике получив символ 0х0D ожидать признак начала посылки 0х55,0хАА, если получен принимать строковые переменные.
Строковые переменные преобразовать в числа.
Дякую за допомогу, але нічого не зрозумів... буду намагатись. Якось простіше ніяк неможна?
Та чому всеж таки не працює Serial.write(buf, len) і Serial.readBytes(buffer, length) ? ну повинно ж передавати декілька байтів по опису...
Неактивний
Та чому всеж таки не працює Serial.write(buf, len) і Serial.readBytes(buffer, length) ? ну повинно ж передавати декілька байтів по опису...
Потому что тип данных не совпадает. Вы массив buff[2] определии как тип int ,
функция принимает тип byte , выше специально привел сноску с описания функции и выделил цветом требуемый тип.
Откройте любой учебник по Си и прочитайте, что такое типы данных, и чем отличается тип byte от int .
Дякую за допомогу, але нічого не зрозумів... буду намагатись. Якось простіше ніяк неможна?
Можно сложнее, простише вряд ли.
Преобразование чисел в строку http://we.easyelectronics.ru/Soft/preobrazuem-v-stroku-chast-1-celye-chisla.html
Так ось і є простіше. Підключив - все працює.
Робочий скетч, передавач:
#include <SoftwareSerial.h>
int valud;
int vallr;
byte buff[2];
int potpinud=0;
int potpinlr=1;
void setup()
{
Serial.begin(9600,SERIAL_8N1);
pinMode(potpinud,INPUT);
pinMode(potpinlr,INPUT);
}
void loop()
{
valud=analogRead(potpinud);
vallr=analogRead(potpinlr);
valud=map(valud,0,1023,1,180);
vallr=map(vallr,0,1023,1,180);
buff[0]=valud;
buff[1]=vallr;
Serial.write(buff,2);
delay(50);
}
приймач:
#include <SoftwareSerial.h>
#include <Servo.h>
Servo servoud;
Servo servolr;
int valud;
int vallr;
byte buff[2];
void setup()
{
Serial.begin(9600,SERIAL_8N1);
servoud.attach(6);
servolr.attach(5);
}
void loop(){
Serial.readBytes(buff,2);
valud=buff[0];
vallr=buff[1];
if (valud>0) {
valud=valud-1;
servoud.write(valud);
delay(20);}
if (vallr>0) {
vallr=vallr-1;
servolr.write(vallr);
delay(20);}
}
![]()
Неактивний
скомпілювало, один серво працює. підключу другий.
а как Вы будете различать где в посылке байты для первой сервы, а где для второй?
Потому что тип данных не совпадает. Вы массив buff[2] определии как тип int ,
функция принимает тип byte , выше специально привел сноску с описания функции и выделил цветом требуемый тип.
Откройте любой учебник по Си и прочитайте, что такое типы данных, и чем отличается тип byte от int .
Та вот читав - недочитав... мозок закипів, починаючий...
хотілось щоб носом ткнули.. але й так допомогло.
Дякую!
Неактивний
wsx пише:скомпілювало, один серво працює. підключу другий.
а как Вы будете различать где в посылке байты для первой сервы, а где для второй?
Ну ось так
valud=buff[0];
vallr=buff[1];
Може теж щось невірно? але воно працює
Неактивний
Так ось і є простіше. Підключив - все працює.
Робочий скетч, передавач:
![]()
![]()
Это не рабочие решение.
Это работает только до тех пор пока индекс буффера передачи совпадает с индексом буфера приемника
Включите/выключите пару раз дурину на приемной стороне и Вы очень быстро получите ситуацию когда приемник будет принимать первым байтом buff[1] ,а вторым buff[1]
первым байтом buff[1] ,а вторым buff[1]
первым байтом buff[1] ,а вторым buff[0] !!!!!
wsx пише:Так ось і є простіше. Підключив - все працює.
Робочий скетч, передавач:
![]()
![]()
Это не рабочие решение.
Это работает только до тех пор пока индекс буффера передачи совпадает с индексом буфера приемника
Включите/выключите пару раз дурину на приемной стороне и Вы очень быстро получите ситуацию когда приемник будет принимать первым байтом buff[1] ,а вторым buff[1]
Спробував пару десятків разів, покищо не було такого...
Але ж і передавач задає порядок:
buff[0]=valud;
buff[1]=vallr;
Serial.write(buff,2);
буду мати на увазі, але поки все ок!
Неактивний
буду мати на увазі, але поки все ок!
Каждый должен наступить свои детские грабли ![]()
Вже наступив
Один раз таке трапилось
Виходить тільки це мені допоможе?:
Свои переменные преобразовать в строку,
Передать: 0х55,0хАА (признак начала посылки) , "строка первой переменой", "строка второй переменой, 0х0D(символ "возврат коретки"окончание посылки)
На приемнике получив символ 0х0D ожидать признак начала посылки 0х55,0хАА, если получен принимать строковые переменные.
Строковые переменные преобразовать в числа.
Можно хоч трохи вразуміліше, як для початківця?
Неактивний
Або якщо просто додати в масив якийсь перший байт та останній, які не співпадають з основними. А в приймачеві розділити по байтах та брати інформаційні після першого "кодового" і останній для перевірки.
Так проканає?
Неактивний
Можно хоч трохи вразуміліше, як для початківця?
можно
байт имеет размерность 8 бит и принимает значения от 0 до 255 или 0x00..0xFF
каждому значению в таблице ASCII соответствует свой символ char
Обычно начало посылки начинают с кодов 0x55 & 0xAA
(в двоичном коде это инверсные последовательности нулей и едениц, 0x55=0b01010101, 0xAA=0b10101010
и при передаче по длинным линиям в коде Грея или по радиоканалу по нем проводят синхронизацию приемника ЮАРТ.)
Чтобы передаваемые данные не совпадали с кодами начала посылки их передают десятичными цифрами ASCII-кодов 0..9 >> 0x30..0x39 ( первая тетрада битов последовательность 0011 , вторая двоичный код цифр)
конец посылки отмечаем или передачей спец.символа Carriage return= 0x0D (возврат коретки) или контрольной суммой.
пример:
byte buff[10];
//две переменных
byte var1=85;
byte var2=170;
//собираем пакет для передачи
buff[0]=0x55; //коды начала
buff[1]=0xAA; //посылки
//переводим числа в строку:
buff[2]=var1/100 + '0'; //сотни
buff[3]=(var1%100)/10 + '0' ; //десятки
buff[4]=var1%10 + '0'; //единицы
buff[5]=var2/100 + '0'; //сотни
buff[6]=(var2%100)/10 + '0'; //десятки
buff[7]=var2%10 + '0'; //единицы
buff[8]=0x0D;// конец посылки
buff[9]=0x00;// NULL признак конца строки.восстановление переменных на стороне приемника
var1 = (buff[2]-'0')*100 + (buff[3]-'0')*10 + (buff[4]-'0');
var2 = (buff[5]-'0')*100 + (buff[6]-'0')*10 + (buff[7]-'0');Або якщо просто додати в масив якийсь перший байт та останній, які не співпадають з основними. А в приймачеві розділити по байтах та брати інформаційні після першого "кодового" і останній для перевірки.
Так проканає?
Serial передає потік даних (stream), сам по собі він безперервний. Ви хочете передавати датаграми, тобто окремі посилки з початком і кінцем.
Є два основних метода:
1. Відокремлювати початок посилки по часу. На передавачі слати дані з певною затримкою між датаграмами (у вас уже є пауза 50 мс). На приймачі, якщо даних не було довше певного періоду, вважати нові дані початком посилки.
2. Відокремлювати початок (та/або кінець) посилки певним маркером. На приймачі сканувати потік на наявність цього маркера. Майте на увазі, що у загальному випадку всередині датаграми також можуть зустрітись дані, які співпадають зі значенням маркера. Щоб уникнути хибної синхронізації бажано додати контрольну суму, хоча б CRC8. При неспівпадінні контрольної суми або маркера кінця посилки не відкидати всі данні, що надійшли, а продовжити сканування з наступного символа після маркера. Правильне рішення цієї задачі не таке просте, як здається на перший погляд.
Кожен метод має свої переваги і недоліки. Можна використовувати і обидва методи одночасно.
Активний