Arduino + два сервопривода + плата зарядки + плата повышающего преобразователя + старый аккумулятор + кучка радиодеталей + кусок фанеры + тумблер = сбылась мечта идиота!
Много текста, для тех, кто любит читать. Много спойлеров, для тех, кто любит читать «по диагонали». Видео, для тех, кто любит видео. Скетч, для тех, кто любит сразу скопировать и запустить «на коленке». Фото, фото, фото. Кот, для любителей котов.
Не совсем дисклеймер
Это мой первый проект, не считая мигания светодиодом, Hello world и т.п. По традиции надо предварительно посыпать голову пеплом, на предмет того, что код далек от совершенства и обязательно ввязаться в спор по этому поводу. А также по поводу частичной скопированности кода у другого проекта, попросить не бить сильно и т.д. Но этого не будет. Код идеален! Потому что работает, нравится мне и сделан для себя.
Интересующимся дам совет: не бойтесь, вступайте в бой, лопатьте горы кода, стройте свои мегапирамиды из разных операторов. Со временем придет и стиль, и утонченность, и идеал.
Пока в памяти контроллера есть место — вам ни чего не страшно. Упретесь в стену — будете оптимизировать. И это тоже развитие. Оно важно.
Большая книга по программированию на С — да, полезно. Но, на первых порах, гораздо полезнее иметь под рукой краткий справочник по командам и держать в закладках несколько ссылок на работу с простыми примерами, библиотеками и т.п., вроде arduino-diy.com, cxem.net или той же Амперки.
И еще, мне очень помогают блок-схемы. Прямоугольник, ромбик, овал. Кто сталкивался — понимает. Делаю сейчас один проект — без блок-схемы никак. Для меня, лучше несколько дней рисовать — перерисовывать на бумаге, чтобы яснее представить себе весь ход работы программы, чем набросать кучу кода и зашиться в нем, т.к. я не обладаю умением охватить весь код силой своей программистской мысли, в виду малого опыта.
Желающим повторить или сделать по-своему – отвечу на все вопросы.
В пустые споры по моей письменной грамотности, стилю, дизайну, коду и нужности изделия и т.п., вступать не буду. Если допустил где-то в обзоре суровый косяк – исправлю или допишу.
Что такое «Самое бесполезное устройство» можете загуглить сами. Я натолкнулся на него случайно здесь. Этот код и был взят за основу, т.к. сам по себе он не запустился, да и сценарии хотелось сделать свои.
Лирическое отступление
Говорят, после сорока, особенно, когда уже очень после сорока, нужно стараться «разминать» мозги. И что изучение языков — лучшая разминка. И не только в таком возрасте. К языкам я не очень, а журнал «Здоровье» как бы советует… В общем, решил я изучить что-то новое для себя. Электроника для меня не в новинку, хоть и забыл я ее уже по большей части, но вот программирование ни когда не было родным. Я его побаивался. Но тут совпало много знаков свыше: журнал «Здоровье», который рекомендует изучить что-то новое, давняя мечта разобраться (хоть немного) в программировании (хоть чего) и youtube, в котором только ленивый не рассказывает про то, как мигать светодиодом с помощью умной платы за доллар с копейками.
До этого я неплохо управлял миганием светодиода с помощью двух транзисторов, конденсатора и пары сопротивлений, но теперь, говорят, это не комильфо. Значит надо быть в тренде.
Очень мне понравилось это бесполезное, в прямом смысле слова, устройство. Увидел – влюбился. Хочу, и всё! Как детская мечта! Но возраст внес свои коррективы. Хочу именно сам сделать, а не купить. Тем более, что в продаже простые устройства. Я же натолкнулся на «интеллектуальное», а таких в продаже я не нашел. Тем более – надо делать!
И опять, Arduino. А я в нем ни бум-бум. Значит надо разбираться. Алиэкспресс в помощь. Начал с Ардуин. Понимал, что путь к изучению будет тернист и жертвы неизбежны. Поэтому заказал пять разных. Пусть себе горят, если что. Отладка схемы происходила на Uno от другого продавца. Но, так как в конечном устройстве оказалась именно эта плата — ее и вынес в заглавие.
Кроме ардуинок, заказал огромное количество всякого разного барахла, как ардуиносовместимого, так и околоардуинового. Здесь расскажу только о том, что пригодилось в данном изделии.
Arduino Pro Mini
Доставка быстрая, упаковка в виде стандартного антистатического пакета и конверта с внутренней пупырчатой поверхностью. Продавец общительный, русскоязычный.
На 328 чипе. 3,3 вольта, 8МГц. Почему именно так? Да по ошибке. Хотел на 5 вольт, 16МГц, а купил эту. Сам виноват. Впрочем, для данного проекта не критично — поставил лишний преобразователь напряжения. Собственно и всё. Огромных отличий от других Mini я не нашел. Фирменное отличие — черный текстолит(?) Из косяков: на плате не работает пин RAW. Но и это не остановило. Хотя, с ним мог бы сэкономить преобразователь на 3,3 В. гребенки не распаяны. Плата сделана добротно.
Почему так кратко? Потому что про этого продавца и его платы уже есть пристрастный обзор (п.18). Желающие найдут легко. Пересказывать его не имеет смысла. Я в комментах к нему тоже вставил свои пять копеек. И с продавцом напереписывался вдоволь.
Сервопривод SG90
Магазин
$3.2 за партию две штуки.
Описывать особенно нечего. Слабый сервопривод со средней точностью позиционирования, которая сильно зависит от скорости перемещения. Зато копеечный. Для привода крышки — с лихвой, для «руки» — на грани возможностей. Для работы нужно 5 вольт, но для управления достаточно и 3,3. Пробовал управлять через преобразователь и напрямую от ардуино — разницы нет. Поэтому преобразователь уровней сэкономил.
Контроллер зарядки и преобразователь питания
Магазин
$2.28 за пять штук. С защитой. Пока заряжается — горит красный светодиод, окончание зарядки — синий светодиод. Обзор здесь был.
Магазин
$0.50 за штуку. Доставка, правда, платная, но я брал в этом магазе еще целый ворох всякой всячины, поэтому доставка не напрягла. USB гнездо выпаял для облегчения веса))) На выходе выдает 5,12 вольт.
Тумблер
Магазин
Брал в оффлайне. Немного туговат. Для облегчения работы сервы, можно было его разобрать, укоротить пружину или заменить на более слабую. Но лень победила. Поставил так. Правда пришлось в коде прописывать добавки к углу поворота сервы на больших скоростях.
TTL преобразователь
Магазин
$1.5 за штуку. Вообще-то я заказал сначала такой. Он в два раза дешевле. Но, по каким-то странным причинам, он пал смертью храбрых. Я так и не понял, чем я его убил. По этой причине проект застопорился на месяц, пока не пришел новый, более продвинутый — не надо давить reset. Это ли не прогресс?
Остальное из запасников
Регулятор напряжения1117Т-3,3V в корпусе TO220, конденсаторы 1500,0х6,3 и 470,0х16, два конденсатора по 0,1 мкФ, белый светодиод и микропереключатель от детской машинки, резистор 220 Ом. Аккумулятор валялся несколько лет без дела. Когда-то я разобрал сгоревший (в прямом смысле) портативный DVD-проигрыватель. Из хорошего там только аккумуляторы и уцелели. Вот один из них и пригодился. Вольтаж 3,7 V, емкости я в маркировке не нашел.
Клеевой пистолет, резинка для денег, два крючка от… лифчика (спасибо супруге. Дорогая, я тебя люблю!), два мебельных шканта, четыре шурупа, четыре клейких отбойника для мебельных дверей, кусок макетной платы, провода, разъемы.
Сложнее обстояло дело с корпусом. Было перепробовано множество коробок. Коробка из пластика от часов Tissot оказалась хрупкой, частично картонной. Но металлические завесы от нее подошли. Одна даже в запасе осталась. Пробовал готовые корпуса от парфюма, коробку от вина, от конфет. В итоге, решил сделать сам.
Качественная фанера, скорее всего березовая, нашлась в детском наборе для выжигания. Ребенок вырос — набор остался. С одной стороны был нанесен рисунок, зато другая сторона выглядела прямо-таки сказать, отлично
Прямо-таки сказать, отлично
В школе у нас был трудовик. В возрасте. В общем и целом — замечательный человек. Его уважали. Когда мы проходили электричество, на примере батарейки, лампочки и ключа, он прочитал нам лекцию.
«В батарейке живет ток. Вот выходит он из плюса и пошел по проводам. Идет, идет, вдруг бац — ключ на пути. И ключ разомкнут. Ток понимает, что дальше идти некуда. Он развернулся и ушел в батарейку. И лампочка не горит. Замыкаем ключ. Ток опять пошел, дошел до ключа, прошел через него, прошел через лампочку и вернулся в батарейку. И лампочка горит, прямо-таки сказать, отлично.» С тех пор и есть такая фраза в лексиконе, как синоним чего-то неожиданно удачного.
Пробовал резать фанеру ножовкой, лобзиком — всё не то. Сколы, которые потом трудно убрать, не ровная кромка. Нашел выход — острый канцелярский нож. Замечательно прорезает половину фанеры с одной стороны, и половину с другой. Потом чуть-чуть наждачкой и всё в ажуре. Зато, при хорошей металлической линейке и постоянно остром лезвии, получается идеальный срез и ровная линия.
Конечно, не без косяков — под наклейкой Hand made скрывается банальная дырка. Просверлил по центру, для тумблера. А серва не влезла. Пришлось смещать тумблер в бок и закрывать дырку. Меня такие трудности не пугают.
Описывать каждый свой чих по изготовлению устройства не вижу смысла. Я опишу некоторые моменты. А уж имеющий руки — да сделает.
Коробку собрал на термоклей. Самого клея не жалел. Держится отменно. Не скрипит, не люфтит. Быстро, дешево и сердито. Да и вообще, как вы заметили, практически всё держится на этом клее. Рекомендую. Очень ускоряет процесс сборки. Микрик выключения приклеил изнутри на левую стенку — виднеется на фото немного.
Крышку прикрепил на завесу.
С завесой намучился. Много экспериментировал. Хотел, чтобы крышка располагалась именно сверху коробки, а не внутри. Даже согнул несколько вариантов скоб-завес из скрепок. По потом вспомнил, что в процессе экспериментов на глаза попались завесы из коробки часов Tissot. Такие добротные, швейцарские (китайские?).
Завеса вещь важная. Ее качество очень важно для общей внешней красоты устройства.
Основной тумблер нужен помягШе, тогда и работать будет полегШе.
Крышку возвращает в закрытое положение резинка. Ее не нужно выбирать очень жесткой — серва справится, но, при центральном расположении завесы, крышку будет перекашивать.
В случае с фанерой, покрытие лаком обязательно — пачкается сильно. Я применил бесцветный цапон-лак. Просто потому, что другого под рукой не оказалось.
Серву, которая управляет «рукой» лучше расположить по-другому. Не сбоку, а спереди от тумблера. Тогда «рука» может быть попроще в изготовлении — г-образная, вместо п-образной. Ширину коробки можно уменьшить и тумблер расположить по центру.
Серву крышки лучше развернуть на 180 градусов, тогда в коде проще будет ассоциировать градусы этой сервы с работой крышки. У меня наоборот, поэтому открывание — это уменьшение градусов. А у «руки» наоборот. И правильнее толкать крышку с той же стороны, с которой ее тянет назад резинка.
Коробку не стоит делать слишком маленькой. Будет не удобно пользоваться. А вот более плоской — допустимо. Коробка легкая и если она будет площе, будет устойчивее. Удобнее будет толкать тумблер пальцем не придерживая коробку.
Из двух кусков макетной платы сделал что-то типа шилда. Ардуину впаял без разъемов. Намертво. Мне не жалко.
А вот всю периферию сделал на разъемах. Так удобнее.
Конденсатор большой емкости по питанию ардуины (3,3 вольт) обязателен. Без него ардуина «виснет».
У меня нет сборочных чертежей устройства. Оно на столько простое, что можно применить много других простых решений, на которых строится вся механика. На видео, как моем, так и других подобных устройств, можно видеть применяемые варианты приводов.
Скетч, предлагаемый автором из ссылки в начале обзора, у меня не запустился. А я еще был не тот профи, каковым являюсь сейчас )))))))
В общем, начал разбираться. В итоге, на основе чужого, сделал свой скетч. Добавил защиту от дребезга. Проще, конечно, было поставить резистор с конденсатором, но уж очень хотелось попрограммировать.
Схема
В принципе, схема понятна из скетча. Но я ее все-таки приведу. Уж простите за качество — как смог. Плату преобразователя взял немного другую — в «анфас» не нашел нужной.
Особых пояснений по скетчу нет. Разве что момент добавления угла поворота на несколько градусов, когда скорость сервы большая. Заметил, что если «рука» выскакивает резко, то она не выключает тумблер. Явно это от низкого качества сервоприводов. Поэтому нужно добавить немного к углу вылета «руки». Подозреваю, что при повторении эти добавочные градусы могут быть у вас другими. Зависит от плеча «руки».
И про программы. Пока шесть программ. Вызываются через генерацию случайного числа. Причем простые программы (0, 1 и 2) вызываются чаще обычных. Всем моим тестировщикам показалось, что более навороченные программы должны быть редким приятным бонусом, тогда появляется некая интрига. Так и сделал.
Для любителей цифр — размер коробки: длина — 150мм, высота — 70мм, ширина — 65мм.
Видео
Youtube сильно ухудшил качество. Если нужно глянуть в оригинале — качайте. 21 МБ.
Здесь в скетч внесено изменение, позволяющее увидеть все шесть программ по очереди, чтобы вы имели представление обо всех. В жизни, как я писал, у них псевдо-случайный порядок.
Зарядка производится через микро-USB обычным зарядником от мобильного. Автономность сильно зависит от частоты использования. Иногда несколько суток, а иногда за день «убиваю».
Напоследок.
Проект, тем не менее, вполне можно дорабатывать и дорабатывать. Можно придумывать новые сценарии. Можно добавить пищалку и озвучить устройство. Например пусть «рычит», типа злится, если время между выключением и включением тумблера очень короткое. Можно, как в исходном проекте, добавить перемещение коробочки в разные стороны.
Можно встроить проверку случая, когда «рука», по какой-то причине, не выключила тумблер (например на холоде, серва не дотягивает до нужного угла совсем немного) и скорректировать угол на один раз, написав специально для такого случая какую-нибудь «нервную» программу выключения. Можно поставить Nano и программировать через USB, не разбирая каждый раз устройство.
Да и вообще – можно сделать аккуратнее. Много всяких можно. Собственно именно этим я и предлагаю заняться тем, кого это заинтересовало.
Может я упустил что-то. Крупноват обзор получился для такого простого устройства. Вот за это могу извиниться.
Теперь кот и спасибо за то, что дочитали до конца.
Летняя фотосессия домашнего мышееда
Планирую купить+79Добавить в избранноеОбзор понравился+123
+232
Извините. Открыл доступ. Закрывал на время редактирования обзора, чтобы добавить в него ссылки на скетч и этот обзор.
А с другой стороны, не специально, но знатно получилось)))
Составляющие движений в функциях move_XX() у вас описаны однотипно. Просятся в отдельную функцию, чтобы у вас были пачки инструкций в одну строку, например,
Ну да. Я после нее взял металлических MG996R по 280 рябчиков на 10Кг… вот это струя. Я сервами кран полуоборотный кручю и… крепеж погнулся. Смотрю, серва по монитору вроде вдижется, потребление есть, а воды нет. Прихожу, а серва двигает, только не кран, а себя. Хватило же дури направляйку с аккуатора двери вывернуть.
А что за кран? или любой крутит? Я тож собираюсь на сервах электрокраны сделать, ибо фабричные с приводами или совсем неприличных денег стоят, или краны на них сомнительные, и где искать в случае поломки самого крана — непонятно
Да, но это клапан а не кран. Я с такими работал — сопротивление потоку они всё-же оказывают намного большее, нежели шаровый кран, жрут много в открытом режиме… Есть «правильные» клапаны — там и тарелка больше, а электромагнит меньше. Там тарелку в закрытом состоянии подпирает сам поток воды, а открывается она снятием этого водяного подпора — маленьким электромагнитом. www.heat-cool.com.ua/img/item/2586/imgdescr.jpg
такой клапан у меня один есть, сопротивление потоку у него меньше, чем у тех, которые с «прямым приводом», но всё-же шаровый это шаровый… один минус — дёргать его надо периодически, чтоб не закисал. Но это уже программная проблема.
И, кстати, по цене. Если надо много (10), то клапаны выйдут в ~22000 р., и жрать будут все разом открытые ампер 10-15. А с сервами выйдет 300 р. серва + кран (300-500), итого 6000-8000 р. если к каждой мелкую тиньку добавить. чтобы получить просто «открыть закрыть» то будет на 100 р. дороже каждая.
Полуоборотный. У меня это краны воды и газа на колонке. При чем если газа еще как-то крутил SG-90, то воду ну ни как. Кручу через коромысла (на крану и на серве по рейке, края рейки соеденены через стальную проволоку где-то 1мм в диаметре.\)
Спасибо за информацию!
Как-то боретесь с закисаниями кранов? Я читал что их надо хотя б раз в пол годика дергать, чтоб не закисли… Хотя если у вас на колонке — то наверное они чаще перемещаются, чем если делать защиту от протеканий на вводе…
Старую конструкцию нет, разобрана. Был смысл такой = — коромысла | — тяги о — кран О — двигатель — - просто заполнитель для движка сайта, он 2 пробела заменяет на 1
======о======
|-------------------|
======O=====
Т.е. верхее коромысло на кран на штуцер закрепляется, нижнее коромысло на двигатель, по края тяги. У меня там краны на 60 градусов, вполне себе хватало.
Сейчас имея 3D принтер все переделано и ось сервы соосна с краном и просто проставка.
Захотелось уточнить, как я понял INPUT_PULLUP — позволяет не ставить на контакт подтягивающий резистор? В статье написано, что это фича новых (с поправкой на 2 года давности) версий ардуины, какая вероятность, что китайские будут это поддерживать?
Это поддерживают все микроконтроллеры Atmega328. НО
1. Встроенные резисторы довольного большого номинала и поэтому довольно помехонеустойчивые
2. Встроенные резисторы, бывают, сгорают. И об этом вы не узнаете.
Так что для надежности лучше проверенные 10к на VCC
молодец, руки из нужного места растут, а я так и не осилил как гуманитарии ардуино, зато «осилил» esp8286, похоже на esp не так много некачественных «изделий», вот отличий от ардуино, хотя сами модули ардуины активно использую, а вот сами платы ардуины не пошли… мистика
ардуин заказывал много и разных, по такой цене удобно получить разведённый контроллер и не травить плату, ну и плюсом стабилизатор и ненужный кварц. Ниразу не попадал на «некачественно», тк не понятно чему там быть некачественному, там же примитив, а контроллер неплохой для простых задач, разве что дорабатывать приходится, если электричество нужно экономить.
соглашусь по аппаратной части может и примитив, а вот программно подбирать библиотеки это не очень, тот же печальный для меня «етхернет» модуль на enc28j60, дешевый вроде, а библиотеку подобрать что бы работало стабильно с моим роутером так и не смог, пришлось покупать «дорогой» и проверенный модуль на Wiznet W5100...(((
ESP-шек заказано три штуки. Мне понравилась идея этих модулей. Есть пара задумок… Но это позже. Пока еще на ардуино не наигрался всё сделано. И три GSM-модуля ждут своего часа. Пока только поигрался с смс-ками и отложил. Торопиться не надо.
А шо? Весьма полезная штука получилась.
Нет ничего полезнее бесполезных вещей, потому как только бесполезные вещи делаются исключительно для души, так что + явно заслужен :)
Плюс за стремление к творчеству.
Замечание за знание английского. «Useless» пишется немного не так, как вы написали.
Лайфхак: пожно написать кирилицей «бесполезная коробка». Ютуб от этого не сломается.
Коли уж начали учить, учите сразу правильно, потому переучиваться сложнее. Начнем:
int pos1door = 70; //начальное положение сервопривода крышки
int pos2door = 30; //конечное положение сервопривода крышки
int pos1hand = 10; //начальное положение сервопривода руки
int pos2hand = 160; //конечное положение сервопривода руки
заменить на #define pos1door 70
Следом использование типов переменных. uint16_t там где это не надо не следует. Есть uint8_t.
Все используемые пины описывайте
#define servo1 5
Так в случае если пин переедет не надо его искать.
Вызываем скажем так:
moveServo(-1);
moveServo(1);
Если добавить ссылки и указатели, то можно еще и для всех серв 1 функцию намаслякать.
Пользоваться random() без randomSeed() не интересно, ибо всегда генерируется одна последовательность по дефолту.
И подтягивающие резистры желательно юзать внешние. Внутренние хоть и есть, но если он умрет — будет не айс…
Не знаю, как в ардуино, в C и C++ размер целочисленных типов не задан жёстко и зависит от платформы. Просто соблюдается соотношение char <= int <= long. Для x86 int 32 бита, x64 — 64 бита. В обычных атмегах int 16 бит, если ничего не путаю. Размер uint16_t будет всегда 16 бит, независимо от компилятора.
«u» в начале означает unsigned — беззнаковый, то есть int8_t может быть от -128 до 127, uint8_t от 0 до 255. Помните стишок про 10 программистов? :)
На самом деле float это не дробное число а целое число с подвижной запятой. Истинное дробное число будет состоять из двух целых чисел представляющих собой числитель и знаменатель, но математика с такими числами становится нетривиальной.
Размерность. msdn.microsoft.com/ru-ru/library/s3f49ktz.aspx
int16_t — 2 байта -32 768 до 32 767
uint8_t — 1 байт — 0 до 255
а скажем int8_t — 1 байт — -128 до 127
int — челых 4 байта! -2 147 483 648 до 2 147 483 647 (он же int32_t)
u — unsigned — т.е. без знака
Скажем мы знаем что положение сервы может меняться только от 0 до 180. Значит нам вполне хватает uint8_t
кратко:
char 1 (8) -128… 127
unsigned char 1 (8) 0… 255
signed char 1 (8) -128… 127
int 2 (16) -32768… 32767
unsigned int 2 (16) 0… 65535
signed int 2 (16) -32768… 32767
short int 2 (16) -32768… 32767
unsigned short int 2 (16) 0… 65535
signed short int 2 (16) -32768… 32767
long int 4 (32) -2147483648… 2147483647
unsigned long int 4 (32) 0… 4294967295
signed long int 4 (32) -2147483648… 2147483647
float 4 (32) 3.4Е-38… 3.4Е+38
double 4 (32) 3.4Е-38… 3.4Е+38
long double 10 (80) 3.4Е-4932… 3.4Е+4932
производные:
int8_t signed char
uint8_t unsigned char
int16_t signed int
uint16_t unsigned int
int32_t signed long int
uint32_t unsigned long int
int64_t signed long long int
uint64_t unsigned long long int
Ну понятно что кого-то надо ткнуть носом, но для общего взора на типы данных в принципе хватает. А уж если голова на плечах есть, то додумать не велика проблема. Если уж не хватает, я бы не советовал садится за программирование. так светиками помигать, чужой код по копипастить, но не более.
Человек про отличия «просто int» спросил, а вы ему «int — челых 4 байта! -2 147 483 648 до 2 147 483 647»,
хотя в AVR GCC int 2 байта -32768… 32767.
Так и представляю: автор написал программу для Arduino Mini, которая ждет условия (переменная_int>100000), спрашивает «почему не работает?» на той же амперке, ему намекают на тип данных, а он — вот же, сам Майкрософт написал!
Иногда лучше отсутствие данных (тогда человек на оф.сайте найдет правильные), чем ложная уверенность в неправильных данных.
Матюк назад. Я виндовый программист привык что int это uint32_t (хотя я вообще делфятник и у меня integer, real и прочие)
А тут для себя понял что чтобы не теряться надо юзать XintN_t и все сразу становится предельно ясно )
культура программирования… Я вот полтора года назад по необходимости достал старый проект. Он лопатил 2 базы и вставлял в 3-ю. В общем на скоряк я обработал ей данные, ушло на это 3 часа. Потом дня 3 я оптимизировал код и менял алгоритмы так что бы работало быстрее и красивее. Итог — те же действия программа делает пол часа. С того момента до сегодняшнего дня я пользовался обработкой порядка 10 раз. Это 25 часов экономии времени. С учетом того что мой рабочий день продуктивный где-то 4-5 часов — я отбил это время,
Мне по наследству достался один проект. В котором в том числе производился расчет ликвидности. Он проходил около 2, потом 3, потом 4х :-).
Добавлением одного (ОДНОГО, КАРЛ!) индекса я свел его до получаса
Тут много нюансов.
Если в табличках 100 записей то это будет отлично работать и без него. даже быстрее чем с ним.
Потом таблички некисло разрастаются и почему-то без него начинает работать хреново :-)
Ну и самый главный принцип: Ты сделай, чтоб хоть как-то работало, потому что завтра вчера нам надо это сдавать :-)
Ну я сам всего ни чего как в С перешел. так что много что сам еще учу. И вот все время хотел почитаь по поводу отличия дефайн и констант. И тут вы меня сподвигли. И вот что я выяснил для себя:
const uint8_t n1=1;
uint8_t n2=2;
n2+=n1 //тут мы обращаемся к ячейке памяти где хранится n1
n2+=n1 //и тут мы обращаемся к ячейке памяти где хранится n1
а с #define… Смотрите сами:
#define n1 1
uint8_t n2=2;
n2+=n1 //тут мы обращаемся к ячейке памяти где хранится 1
n2+=n1 //а тут мы обращаемся к другой ячейке памяти где хранится 1
Выходит define ложная катавасия в плане экономии ресурсов
Плюс (нам это не грозит) при отладке скажем при #define test 123 в случае если будет проблема с этими 123, то отладчик покажет 123, а не test и получистя не будет известно от куда взялось это 123. Т.е. скажем когда
uint8_t n1=0;
#define test 5/n1
мы не будем знать где произошло это деление на 0 точно. В таком случае в C++ (в С нет) inline функции. Блин… я программист, меня год учили C++, а я сдал последнюю лабу за 1 месяц и не ходил на занятия целый курс… Я много пропустил )
Директивы препроцессора это страшная вещь… :)
Если интересно, вот большая статья про них.
Удивительно, но в плюсах тоже частенько можно встретить, причём не только в роли #include guard. Последний особенно впечатливший меня пример — это реализация исключений в Poco. Нет, я всё понимаю, но взять и настрогать классов дефайнами…
Насчёт памяти вы правы, хотя мне кажется, что компилятор должен уметь оптимизировать такие вещи.
Ну я читал что вроде бы у стокового компилятора не все гладко с этим. Хотя я пройдясь по uint16_t и переписав местами const в #define, местами прикрутив макросы вместо функций (теперь знаю что надо делать inline), и отказавшись от либо явного указания переменных с плавающими точками либо вообще от их использования с 80% памяти съехал до 50. А глобальным перемнным вообще норм, с 80% до 30%. Долго тужился с отказом от string, но выходило слишком громоздко и нудно (utf8...) и обратно ввел и все что я наоптимизировал все ушло в string (
«1» нигде не хранится, в том смысле в каком это понимаете вы — эта константа будет непосредственно в коде.
А вот если у нас n1 — это const uint8_t, тогда в коде у нас будет другая константа: адрес той ячейки памяти, в которой лежит значение «1». И добавится лишняя операция — чтение по данному адресу.
Хотя, компилятор может и соптимизировать это, сведя ситуацию к предыдущей. Кстати, интересный вопрос, каковы дефолтные опции компилятора в среде ардуино. Надо б посмотреть… :)
В микроконтроллере гарвардской архитектуры (в т.ч. ATmega328P в Arduino Mini) не одно адресное пространство. Отдельно — память команд, отдельно — данных.
Константа непосредственно в коде (#define) — инструкция LDI www.gaw.ru/html.cgi/txt/doc/micros/avr/asm/ldi.htm — занимает 2 байта памяти программ и исполняется за 1 цикл, константа в оперативной памяти (const) — LDS www.gaw.ru/html.cgi/txt/doc/micros/avr/asm/lds.htm — 4 байта и 3 цикла + место в памяти данных под эту константу, константа в памяти программ (PROGMEM const) — LDM www.gaw.ru/html.cgi/txt/doc/micros/avr/asm/lpm.htm — 2 байта и 3 цикла (и еще команда загрузки в Z) + занимаемая под константу память программ. Так что если константа не более 255 — #define оптимален и с точки зрения экономии памяти программ, и памяти данных, и быстродействия.
в программах для большого брата препроцессор выглядит устаревшим и не желателен к использованию (в первую очередь из-за непрозрачного сопровождения кода), в атмегах — он вполне работает и позволяет экономить ресурсы
По-моему, вы зря лезете в эти дебри. Если уж так важны эти такты, то надо отказываться от ардуино и писать под голый контроллер на ассемблере, которого я, например, не знаю и не горю особым желанием учить.
Ну есть культура написания кода. Т.е. какие-то правила которые позволяют код сделать изначально оптимальным и красивым. Вот эти дебри и позволяют где-то заюзать define, а гдето inline, а где-то функцию, а где-то константу. Вот сегодня оптимизировал код одного паренька. В принципе после оптимизатора мой код выигрывал у оригинала порядка 30% программного кода и 50% глобальных переменных. СРазу я начал переписывать (читай исправлять) код, ушло на это 2 часа и я прошел только 1/3 всего кода. Потом я плюнул на все, и написал весь код заного учитывая мои познания. Ушло порядка 2-х часов, но результаты я вам уже представил.
Константы это числовые подстановки, константа может содержать только число. А дефайн вещь более гибкая и мощная. Это текстовая подстановка на уровне исходника. Использование дефайна в качестве константы это лишь частный случай, надо отдавать себе отчёт что это будет просто подстановка в исходнике, а по нему уже ПОТОМ пройдётся компилятор. Чтобы не забыть что есть что к дефайнам обычно добавляют подчеркивание в начале имени, вроде как грязный гаечный ключ лежащий в машинном отделении.
Мы уже установили это. Вопрос стоял в другом — оптимизация. Скажем используя const uint_t n=5; мы использовали 2 байта в global variables, в то время как #define n 5 в именно кодинге ни чем не отличается кроме того что количество циклов обращений сводится к 1 и не занимает места в памяти кроме своего места в коде.
имея опыт творчества и читая текст, вижу между строк сколько трудов, неудач, матов и, конечно, радости от работающей поделки пришлось на долю автора.
однозначно плюс…
а главное, когда пишешь — вроде и писать особо нечего — все очевидно (поле граблей после изготовления готового устройства — кажется только твоим, и у других, опять же ощущение, его не возникнет)
круто было бы найти список использованного железа, которое есть на али, и выкройки корпуса и «пальцев» для сервов, чтобы повторить такую коробку, не заморачиваясь.
Всё «железо» в статье. Добавить можно разве дюпон-соединители. Брал здесь. Но их использовано несколько штук, и заказывать много не стОит. Также использованы разъемы от старого системника (female) для светодиода и тумблера. Клей и пистолет куплены так давно, что даже не помню где. Но не на Али.
Выкройка корпуса проста — простые формы. Размер не проектировался специально. Были листы фанеры, формата 200х315 толщиной 3 мм. Я его обрезал до размера 150х315 мм. Потом покромсал на полоски двух ширин: одна полоса шириной 59мм и четыре полосы шириной 64мм. Верхняя, нижняя и боковые крышки абсолютно одинаковые, 150х64мм. Зад и перед вырезаны из одной полосы, их размер 64х59мм. Крышка формируется легко — распиливаем одну полосу ножовкой по металлу. Ее толщина и есть зазор между подвижной и неподвижной крышками. Толкатель крышки в моем примере имеет размер 28х9мм с закругленными углами. «Руку» вырезал так. Приклеил серву на тонкий двусторонний скотч. Взял кусок толстого картона и накидал выкройку. Привинтил ее винтами и правил ножницами. Без подключения сервы. Просто двигал рукой. Центр отверстия для тумблера расположен на расстоянии 21мм от края, из-за которого вылетает рука. Размер руки нужно подбирать исходя из высоты тумблера и хода его рычажка.
У меня «рука» имеет такие размеры:
Поверьте, из картона заготовка вырезается на раз.
В моем случае, я ошибся с расположением сервы. Правильнее было бы установить не сбоку, а перед тумблером, вытянув ее вертикально. Тогда можно упростить «руку» до г-образной, и сделать ее тоньше.
Если захотите прикрепить какую-то нашлепку, для пущего дизайна, обязательно просверлите под ней дырку. Тогда у этой детали появится еще и сугубо практическое назначение.
На моторе с редуктором вы могли использовать ик датчик расстояния? Или рандомное время когда вылезет рука. Или движения туда/сюда? Ардуина — это всего лишь МК с обвязом готовая к употреблению и способная в любой момент времени быть прошита новой прошивки без мучений с фьюзами и прочим. Так что на одном моторе вы бы ни уехали.
Я знаю, что такое Ардуино. И без неё сделал уже больше десятка самых бесполезных устройств на подарки, в которых не было Ардуино, только моторчик, который работает от батареек
Кто, интересно, придумал название «Самое бесполезное устройство»? Раз оно развлекает пользователя — оно уже всяко не бесполезное! Сам Клод Шеннон назвал это своё изобретение «Абсолютной машиной». Говорят, что когда он показал свою абсолютную машину Артуру Кларку, у того случился культурный шок… ;)
Подскажите, пожалуйста, будет ли достаточно производительности и разъемов Atmega328 для чтения NMEA-сообщений из GPS-модуля и записи их на карту памяти? Сколько примерно будет потреблять, есть ли разница между 5-и и 3-вольтной версиями?
Находил описание, как писать на карту, но хотелось бы изучить девайс последовательно. С чего начать, желательно на русском языке.
Хватит с головой. GPS-модуль будет в 10 раз больше потреблять.
А 3 или 5В версия надо подбирать под GPS-модуль чтобы потом не добавлять согласование уровней. Скорей всего нужно будет 3-х вольтовую версию.
Большое спасибо автору, очень крутая статья. Загорелся покупкой ардуины и изготовлением такого девайса.
Может знающие подскажут, что лучше взять — arduino nano или micro?.. И еще вопрос — выдержат ли эти ардуино мощность серв? Это нужно в даташите на ардуино смотреть?
Если выбирать между нано и микро, то лучше микро, но они и дороже. Если гнаться за рекордом по цене, то лучше про-мини на АТМеге168 вместо 328, для многих применений этого вполне достаточно. ;) По мощности: насколько помню, нагрузочная способность ардуины — до 40мА на ногу и то желательно поменьше нагружать, поэтому для сервоприводов нужны драйверы.
Вообще сервам драйверы не нужны, они управляются слаботочкой а большой ток потребления у них по шине питания что уже собственно выходов контроллера не касается и проблемы могут возникнуть только при неудачной разводке шин питания.
Забавная штука! Дочка впечатлилась вашим творением — пришлось делать заказ на ардуино и сервы. Правда в ардуино я полный ноль. А TTL преобразователь для чего? Программировать?
А с другой стороны, не специально, но знатно получилось)))
Продраться сквозь не смог…
В коде в setup() напрашивается INPUT_PULLUP
Составляющие движений в функциях move_XX() у вас описаны однотипно. Просятся в отдельную функцию, чтобы у вас были пачки инструкций в одну строку, например,
step(from;to;increment);
step(from;to;increment);
step(from;to;increment);
так дебажить проще. Одинаковые значения (например, паузы между движениями) — в константы надо бы убрать, чтобы можно было поправить в одном месте.
Сезон отработали 8 штук — все норм.
такой клапан у меня один есть, сопротивление потоку у него меньше, чем у тех, которые с «прямым приводом», но всё-же шаровый это шаровый… один минус — дёргать его надо периодически, чтоб не закисал. Но это уже программная проблема.
И, кстати, по цене. Если надо много (10), то клапаны выйдут в ~22000 р., и жрать будут все разом открытые ампер 10-15. А с сервами выйдет 300 р. серва + кран (300-500), итого 6000-8000 р. если к каждой мелкую тиньку добавить. чтобы получить просто «открыть закрыть» то будет на 100 р. дороже каждая.
Как-то боретесь с закисаниями кранов? Я читал что их надо хотя б раз в пол годика дергать, чтоб не закисли… Хотя если у вас на колонке — то наверное они чаще перемещаются, чем если делать защиту от протеканий на вводе…
======о======
|-------------------|
======O=====
Т.е. верхее коромысло на кран на штуцер закрепляется, нижнее коромысло на двигатель, по края тяги. У меня там краны на 60 градусов, вполне себе хватало.
Сейчас имея 3D принтер все переделано и ось сервы соосна с краном и просто проставка.
1. Встроенные резисторы довольного большого номинала и поэтому довольно помехонеустойчивые
2. Встроенные резисторы, бывают, сгорают. И об этом вы не узнаете.
Так что для надежности лучше проверенные 10к на VCC
наигралсявсё сделано. И три GSM-модуля ждут своего часа. Пока только поигрался с смс-ками и отложил. Торопиться не надо.Типа, работать не хочет.
потребление батареи
Нет ничего полезнее бесполезных вещей, потому как только бесполезные вещи делаются исключительно для души, так что + явно заслужен :)
Так что автору есть куда стремиться =)
+ однозначно.
Замечание за знание английского. «Useless» пишется немного не так, как вы написали.
Лайфхак: пожно написать кирилицей «бесполезная коробка». Ютуб от этого не сломается.
спасибо за желание начать разбираться!
заменить на #define pos1door 70
Следом использование типов переменных. uint16_t там где это не надо не следует. Есть uint8_t.
Все используемые пины описывайте
Так в случае если пин переедет не надо его искать.
Описываем как отдельные функции:
Вызываем скажем так:
moveServo(-1);
moveServo(1);
Если добавить ссылки и указатели, то можно еще и для всех серв 1 функцию намаслякать.
Пользоваться random() без randomSeed() не интересно, ибо всегда генерируется одна последовательность по дефолту.
И подтягивающие резистры желательно юзать внешние. Внутренние хоть и есть, но если он умрет — будет не айс…
«u» в начале означает unsigned — беззнаковый, то есть int8_t может быть от -128 до 127, uint8_t от 0 до 255. Помните стишок про 10 программистов? :)
А вообще все эти _t нафиг не нужны, пока вас не беспокоят проблемы портируемости и кроссплатформенности.
int16_t — 2 байта -32 768 до 32 767
uint8_t — 1 байт — 0 до 255
а скажем int8_t — 1 байт — -128 до 127
int — челых 4 байта! -2 147 483 648 до 2 147 483 647 (он же int32_t)
u — unsigned — т.е. без знака
Скажем мы знаем что положение сервы может меняться только от 0 до 180. Значит нам вполне хватает uint8_t
habrahabr.ru/post/156593/ — вот популярно
тут кратко и таблично применительно к AVR (а значит к ардуино):
digitalchip.ru/skolko-vesyat-tipyi-dannyih
кратко:
char 1 (8) -128… 127
unsigned char 1 (8) 0… 255
signed char 1 (8) -128… 127
int 2 (16) -32768… 32767
unsigned int 2 (16) 0… 65535
signed int 2 (16) -32768… 32767
short int 2 (16) -32768… 32767
unsigned short int 2 (16) 0… 65535
signed short int 2 (16) -32768… 32767
long int 4 (32) -2147483648… 2147483647
unsigned long int 4 (32) 0… 4294967295
signed long int 4 (32) -2147483648… 2147483647
float 4 (32) 3.4Е-38… 3.4Е+38
double 4 (32) 3.4Е-38… 3.4Е+38
long double 10 (80) 3.4Е-4932… 3.4Е+4932
производные:
int8_t signed char
uint8_t unsigned char
int16_t signed int
uint16_t unsigned int
int32_t signed long int
uint32_t unsigned long int
int64_t signed long long int
uint64_t unsigned long long int
__int16 работает в Visual Studio 2015 — там так и написано
табличка для тех кто там пишет, а остальные, в том числе и автор обзора, пишут не там
у майкрософт свои стандарты — отличающиеся от других, туда нужно ссылаться если разработка идет в той среде
хотя в AVR GCC int 2 байта -32768… 32767.
Так и представляю: автор написал программу для Arduino Mini, которая ждет условия (переменная_int>100000), спрашивает «почему не работает?» на той же амперке, ему намекают на тип данных, а он — вот же, сам Майкрософт написал!
Иногда лучше отсутствие данных (тогда человек на оф.сайте найдет правильные), чем ложная уверенность в неправильных данных.
А тут для себя понял что чтобы не теряться надо юзать XintN_t и все сразу становится предельно ясно )
Добавлением одного (ОДНОГО, КАРЛ!) индекса я свел его до получаса
Если в табличках 100 записей то это будет отлично работать и без него. даже быстрее чем с ним.
Потом таблички некисло разрастаются и почему-то без него начинает работать хреново :-)
Ну и самый главный принцип: Ты сделай, чтоб хоть как-то работало, потому что
завтравчера нам надо это сдавать :-)const uint8_t n1=1;
uint8_t n2=2;
n2+=n1 //тут мы обращаемся к ячейке памяти где хранится n1
n2+=n1 //и тут мы обращаемся к ячейке памяти где хранится n1
а с #define… Смотрите сами:
#define n1 1
uint8_t n2=2;
n2+=n1 //тут мы обращаемся к ячейке памяти где хранится 1
n2+=n1 //а тут мы обращаемся к другой ячейке памяти где хранится 1
Выходит define ложная катавасия в плане экономии ресурсов
Плюс (нам это не грозит) при отладке скажем при #define test 123 в случае если будет проблема с этими 123, то отладчик покажет 123, а не test и получистя не будет известно от куда взялось это 123. Т.е. скажем когда
мы не будем знать где произошло это деление на 0 точно. В таком случае в C++ (в С нет) inline функции. Блин… я программист, меня год учили C++, а я сдал последнюю лабу за 1 месяц и не ходил на занятия целый курс… Я много пропустил )
Если интересно, вот большая статья про них.
Удивительно, но в плюсах тоже частенько можно встретить, причём не только в роли #include guard. Последний особенно впечатливший меня пример — это реализация исключений в Poco. Нет, я всё понимаю, но взять и настрогать классов дефайнами…
Насчёт памяти вы правы, хотя мне кажется, что компилятор должен уметь оптимизировать такие вещи.
uint8_t n2=2;
n2+=n1;
«1» нигде не хранится, в том смысле в каком это понимаете вы — эта константа будет непосредственно в коде.
А вот если у нас n1 — это const uint8_t, тогда в коде у нас будет другая константа: адрес той ячейки памяти, в которой лежит значение «1». И добавится лишняя операция — чтение по данному адресу.
Хотя, компилятор может и соптимизировать это, сведя ситуацию к предыдущей. Кстати, интересный вопрос, каковы дефолтные опции компилятора в среде ардуино. Надо б посмотреть… :)
Константа непосредственно в коде (#define) — инструкция LDI www.gaw.ru/html.cgi/txt/doc/micros/avr/asm/ldi.htm — занимает 2 байта памяти программ и исполняется за 1 цикл, константа в оперативной памяти (const) — LDS www.gaw.ru/html.cgi/txt/doc/micros/avr/asm/lds.htm — 4 байта и 3 цикла + место в памяти данных под эту константу, константа в памяти программ (PROGMEM const) — LDM www.gaw.ru/html.cgi/txt/doc/micros/avr/asm/lpm.htm — 2 байта и 3 цикла (и еще команда загрузки в Z) + занимаемая под константу память программ. Так что если константа не более 255 — #define оптимален и с точки зрения экономии памяти программ, и памяти данных, и быстродействия.
имея опыт творчества и читая текст, вижу между строк сколько трудов, неудач, матов и, конечно, радости от работающей поделки пришлось на долю автора.
однозначно плюс…
а главное, когда пишешь — вроде и писать особо нечего — все очевидно (поле граблей после изготовления готового устройства — кажется только твоим, и у других, опять же ощущение, его не возникнет)
Посмотрите лучше в сторону гексаподов:
Плюсанул
Выкройка корпуса проста — простые формы. Размер не проектировался специально. Были листы фанеры, формата 200х315 толщиной 3 мм. Я его обрезал до размера 150х315 мм. Потом покромсал на полоски двух ширин: одна полоса шириной 59мм и четыре полосы шириной 64мм. Верхняя, нижняя и боковые крышки абсолютно одинаковые, 150х64мм. Зад и перед вырезаны из одной полосы, их размер 64х59мм. Крышка формируется легко — распиливаем одну полосу ножовкой по металлу. Ее толщина и есть зазор между подвижной и неподвижной крышками. Толкатель крышки в моем примере имеет размер 28х9мм с закругленными углами. «Руку» вырезал так. Приклеил серву на тонкий двусторонний скотч. Взял кусок толстого картона и накидал выкройку. Привинтил ее винтами и правил ножницами. Без подключения сервы. Просто двигал рукой. Центр отверстия для тумблера расположен на расстоянии 21мм от края, из-за которого вылетает рука. Размер руки нужно подбирать исходя из высоты тумблера и хода его рычажка.
У меня «рука» имеет такие размеры:
Поверьте, из картона заготовка вырезается на раз.
В моем случае, я ошибся с расположением сервы. Правильнее было бы установить не сбоку, а перед тумблером, вытянув ее вертикально. Тогда можно упростить «руку» до г-образной, и сделать ее тоньше.
Если захотите прикрепить какую-то нашлепку, для пущего дизайна, обязательно просверлите под ней дырку. Тогда у этой детали появится еще и сугубо практическое назначение.
В принципе, моторы у меня тоже с Али, могу и обзор запилить:)
Находил описание, как писать на карту, но хотелось бы изучить девайс последовательно. С чего начать, желательно на русском языке.
А 3 или 5В версия надо подбирать под GPS-модуль чтобы потом не добавлять согласование уровней. Скорей всего нужно будет 3-х вольтовую версию.
Может знающие подскажут, что лучше взять — arduino nano или micro?.. И еще вопрос — выдержат ли эти ардуино мощность серв? Это нужно в даташите на ардуино смотреть?
А если бы еще динамик с «рычанием» «гавканьем» и жалобными звуками добавили… ребетенка было бы не оторвать)))
Если бы наш дед соорудил такую штуку своему внуку… то привил бы страсть к такому творчеству надолго)))
Только зарегистрированные и авторизованные пользователи могут оставлять комментарии.