Помогите. Срочно нужен эластоллан 1175а. Полиуретан
Уважаемые пикабушники. Выручайте. Срочно нужен Elastollan 1175A. Возьму большой объем!
Уважаемые пикабушники. Выручайте. Срочно нужен Elastollan 1175A. Возьму большой объем!
Здравствуйте, коллеги. Сегодня мы с вами познакомимся с организацией работы при ограниченных ресурсах. Рассмотрим что такое "Очередь" как она работает, реализуем пару вариантов очередей, затронем динамическое выделение памяти в ПЛК.
Я пишу эту статью и программирую в режиме реального времени, т.е. у меня есть задача и я ее решаю. Все свои мысли и попытки(за исключением очень глупых) я записываю в эту статью так что приятного чтения.
ТЕОРИЯ
Очередь - это абстрактный тип данных, в котором доступ организуется по принципу FIFO(first in, first out) - первый пришел, первый ушел. Имеет два метода: добавить в очередь и достать из очереди. Данные которые мы достали из очереди - удаляются.
Согласно теории есть два метода реализации - массивом и односвязным списком. В большинстве ПЛК второй способ невозможен.
РЕАЛИЗАЦИЯ
Начнем с малого. Реализуем очередь для одного типа данных. Например, мы будем вырезать отверстия. По нужным координатам нужного диаметра.
Создадим необходимы DUT. В структуре у нас хранятся координаты по X,Y и диаметр отверстия.
Теперь организуем список необходимых операций, которые нам понадобятся. Всего их три: Ожидание, Добавить элемент, Достать элемент. Для такого типа операций я использую перечисление и CASE в реализации. Мне просто так удобнее. Можно с помощью флагов, триггеров, IF'ов, но я делаю так.
Теперь работаем с FB. Пытаемся уместить все в одном месте. Нам потребуется на вход:
1) Указатель на Глобальный UDT для чтения
2) Указатель на UDT куда будем писать.
(*тут я думаю как сделать команды*)
3) Команды управления
На выход у нас получается:
1) Сигнал об успешной записи в очередь
2) Сигнал об успешном чтении из очереди
3) Очередь пустая
4) Очередь полная
5) В работе
Пункты 3 и 4 можно объединить в ошибку и сделать какой-нибудь код для нее, но я не буду.
Под капотом:
1) Массив элементов( для теста 10)
2) Индекс старта
3) Индекс окончания
ЛОГИКА РАБОТЫ
Логика легка и проста. читаем из элемента под индексом iStart. Считали прибавили 1. Записываем в индекс iEnd. Записали прибавили единицу.
Дошли до конца, скинули счетчик. Пошли по кругу. Всего может быть 10 элементов. Если индексы пересекаются, то это неопределенность. Либо очередь полная, либо пустая. Так что этот вопрос продеться контролировать.
При старте реализации было обнаружено, что значения пустой и полной очереди необходимо обрабатывать внутри кода. Есть два три выхода: переносим в In_Out, переносим указателями в In, создаем в области VAR и во время WAIT копируем значения.
Я выбираю третий вариант с дублированием.
Добавил в Output переменную Error.
ИТОГ
Я опять пожалуюсь, что вставлять и сюда код очень проблематично так что ссылка на PDF документ на Яндекс.Диск с листингом.
Ну и небольшая видео демонстрация как там все происходит.Спасибо за внимание. Если что стучите в комментарии, если что-то очень личное, то на почту info@engcore.ru.
Добрый день всем. Сей шкаф нужно подключить к двум насосам грундфос( в наличии они). Голова здесь CU361(0 или ноль). Вторичный IO351B. С местом подключения трёхфазных насосов разобрался. А вот какой из двух термодатчиков (160 или 170°с) которые встроены в насос нет. И не понимаю куда их подсадить. К башке или вторичному? И ещё три поплавка нужно подсадить. На гребёнке есть и с головы и со вторички выводы. Если что. Добавлю фоток. Пасспорт утерян. С 12 года не подключеный стоит.
Пикабутяне, вопрос у меня. Хочу поставить на трубы горячей воды (2 шт) в квартире датчики температуры (термодатчик) чтобы фиксировать температуру в постоянном режиме. При этом подключить к компьютеру. По хорошему чтобы сертифицированное было всё и по адекватной цене. Как это можно сделать? В интернете навалом всяких датчиков, но не понятно что нормальное а что нет. Заранее спасибо.
Ааааааааааа.... Несколько дней прошло с зудящей мыслью, а что это за фрукт такой ANY_DB. И вот наконец-то теоретическое послание о том что такое ANY_DB и попытки разобраться, а зачем он нужен(я так и не понял, но об этом после).
Немного неофициальной теории
В целом это еще один инструмент для унификации процессов и написании мега универсальных ФБ. Особенность того, что ANY_DB хранит номер DB и может быть использован до того как мы создали дб. В ANY_DB хранится номер любой БД(Ох если все было так просто).
Вы не сможете записать в параметр ANY_DB DB с резервированием памяти
И так что мы можем точно писать в ANY_DB:
1)Экземпляры UDT и системных типов
2)Экземпляры FB
3)DB которые были созданы как массив.
Из этого списка пункты 1 и 3 имеют возможность конвертирования в Variant. В целом есть две команды:
VARIANT_TO_DB_ANY
DB_ANY_TO_VARIANT
И больше с этим типом данных нельзя сделать НИ-ЧЕ-ГО!
Как ANY_DB в VARIANT превратить и что с этим дальше делать.
В целом после столь скудной теоретической части давайте фантазировать.
ANY_DB мы можем конdертить в variant.
А дальше там скармливать что хочешь и куда хочешь, у нас же есть REF_TO и ?=
В коде это будет выглядеть следующим образом:
Ну и если подняться выше, то можно узнать, что на вход ANY_DB я подаю Data_Block_2
Тут для закрепления требуется какая-то задачка, но на ум пока ничего толкового не приходит. Разве что какая-то фантастическая сборная линия с тремя операциями по координатам. Но если вдруг у вас появились идеи как это куда-то приделать или вы использовали где-то, то пожалуйста, поделитесь информацией.
Появилось желание потыкать более подробно ПЛК 1500 серии от Siemens. И немного разобрать что там интересного есть, как с этим работать и понять для чего это надо. И первыми под небольшой обзор попала такая группа данных как pointer(указатель), которых в s7-1500 аж 4 вида. Сейчас хочу быстро пробежимся по каждому и немного поработать с References
Типы данных в ПЛК.
Начнем с теории. Типы данных - определяют свойства данных, например, размер в памяти или их представление. У Siemens есть:
Элементарные типы данных(bool, int, real, char)
Сложные типы данных (DT, String, Array, Struct)
UDT - типы определяемые пользователем
УКАЗАТЕЛЬ
Тип параметров
Системные типы данных
Типы данных оборудования
Ну и если с верхней частью списка все более менее понятно, исходя из начальных уроков по программированию. То где-то с указателей начинается магия. Однако все не так просто.
Немного об оптимизированном доступе
У контроллеров S7-1200/1500 имеется возможность оптимизированногоИ теперь если просто, то в оптимизированном блоке нет доступа по адресу. Там происходит магия, данные сортируются так как надо, теперь тебе не надо играть в тетрис, когда у тебя все бинарные значения должны быть в конце. Но от этого мы получаем новые виды указателей.
хранения данных. В оптимизированных блоках, все теги автоматически
сортируются по их типу данных. Данный метод позволяет минимизировать
промежутки между тегами, таким образом такие теги оптимизированы по
времени доступа для процессора.
(c)Базовое системное руководство 11/2015Руководство по программированию
S7-1200/S7-1500
УКАЗАТЕЛИ
В данную группу входят следующие ребята:
- References
- VARIANT
- POINTER
- ANY
Для наглядности где какие указатели могут быть использованы вне 1500 контроллеров ловите табличку.
References
References - это тэг, который указывает на область памяти. вы указываете тип данных, на который он ссылается, является безопасным по типу. Гарантируется, что либо она принадлежит к конкретному типу данных, либо ей присваивается NULL.
References у нас указывают только на конкретный тип и могут быть прочитаны или записаны через разыменовывание. ТОЛЬКО ДЛЯ ОПТИМИЗИРОВАННЫХ БЛОКОВ,
VARIANT
Это тэг, который указывает на различные типы данных, но не может указывать на экземпляры. Операнд типа данных VARIANT не занимает места в блоке данных экземпляра или в рабочей памяти. Однако он будет занимать место в памяти ЦП.
Тег типа VARIANT - это не объект, а ссылка на другой объект. Отдельные элементы типа VARIANT могут быть объявлены только для формальных параметров в блочном интерфейсе функции в разделах VAR_IN, VAR_IN_OUT и VAR_TEMP. По этой причине его нельзя объявить в блоке данных или в статическом разделе интерфейса блока функционального блока, например, потому что его размер неизвестен. Размер объектов, на которые имеются ссылки, может измениться.Ну и много других вещей. Вот тут уже можно себе выстрелить немножко в ножку, но радует, что мы не можем просто читать и записывать Variant.
(с) Перевод гуглом справки от Сименса
ПРАКТИКА
Начнем с референсов)) Объявить данный тип мы можем в FC,FB,OB, но в разных частях интерфейса
- FC: Input, Output, Temp, Return
- FB: Temp
- OB: Temp
Объявление происходит конструкцией REF_TO, присваивание значений командой REF();
А дальше мы пойдем немного усложнять задачи. И чуток поработаем в связке VARIAN и REF. Придумаем себе несуществующую задачу, которая гласит, что у нас есть функция, на вход которой могут подаваться различные UDT.
В нашем случае следующего вида:
И нам надо записать туда значения. в случае udt_2pointж 5.0, а в случае udt_3point -4.
Из всего что написано выше, что стоит понять функция TypeOf - возвращает тип тэга Variant.
Конструкция ?= пытается присвоить REF тэг Variant. Так как это слегка удобнее. Если присвоение провалилось, то тэг REF будет NULL.
Теперь на вход подаем udt_2point.
Тут должен быть примерно какой-то вывод, но в целом даже подобную конструкцию можно использовать для создания какого-нибудь максимально универсального блока с проверкой типов данных и относительной безопасностью.