Имитация работы автоматической коробки передач
В данном примере мы разберём инструмент Chart и реализуемую им логику конечных автоматов, основанную на принципах автоматического переключения передач в зависимости от скорости.
На упрощённом уровне рассмотрим различия в работе алгоритмов двух типов – с применением состояний и с использованием узлов переходов.
В начале демонстрации объявим вспомогательную функцию для запуска модели и перейдём к анализу реализованных алгоритмов.
# Подключение вспомогательной функции запуска модели.
function run_model( name_model)
Path = (@__DIR__) * "/" * name_model * ".engee"
if name_model in [m.name for m in engee.get_all_models()] # Проверка условия загрузки модели в ядро
model = engee.open( name_model ) # Открыть модель
model_output = engee.run( model, verbose=true ); # Запустить модель
else
model = engee.load( Path, force=true ) # Загрузить модель
model_output = engee.run( model, verbose=true ); # Запустить модель
engee.close( name_model, force=true ); # Закрыть модель
end
sleep(5)
return model_output
end
На верхнем уровне модель содержит два алгоритма перключения передач. Входы для обоих чартов идентичны. В таблицах заданы пороговые значения для скорости, при которых необходимо либо повысить передачу, либо понизить её, и само значение текущей скорости, заданное синусоидой.

Далее рассмотрим каждый Chart в отдельности. Верхний блок использует логику узлов и в зависимости от скорости выдаёт параметр сдвига передачи +-1, что позволяет выполнить повышение или понижения передачи, которое хранится в петле. Начальное значение передачи в обоих случаях равно 1.

Во втором случае в зависимости от скорости мы переключаемся между различными состояниями, в которых задаётся подходящая передача под входной скоростной режим.

Теперь запустим саму модель и проанализируем залогированные значения.
run_model("car") # Запуск модели.
Считываем из simout залогированные данные.
В данном случае:
v – это скорость автомобиля;
shift – параметр сдвига для алгоритма коробки передач, реализованной с использованием узлов;
gear_node – значения передачи для Chart с использованием узлов;
gear_state – значения передачи для Chart с состояниями.
v = simout["car/Bias.1"];
v = collect(v);
shift = simout["car/ShiftChart.shift"];
shift = collect(shift);
gear_node = simout["car/gear"];
gear_node = collect(gear_node);
gear_state = simout["car/GearChart.gear"];
gear_state = collect(gear_state);
Теперь перейдём к анализу полученных данных. Начнём со скорости. Как мы видим из графика, приведённого ниже, скорость задана синусоидой в диапазоне от 0 до 120. Начальное значение скорости не нулевое. Оно задано с целью наглядной демонстрации различий алгоритмов с применением узлов и состояний.
plot(v.time, v.value)
Далее проанализируем систему с применением узлов. Как мы видим, начальное значение данной системы на первом шаге – не первая передача, а вторая. Это связано с тем, что узел сразу сообщил о приращении счётчика.
plot(gear_node.time, gear_node.value)
plot!(shift.time, shift.value)
Теперь обратим внимание на систему, реализованную на основе состояний. Начальное значение передачи равно 1. Это связано с тем, что в случае с состояниями при отсутствии логики инициализации мы начнём с входного состояния и далее будем проходить максимум по одному состоянию за такт.
plot(gear_state.time, gear_state.value)
Теперь сравним значения, рассчитанные различными системами имитации работы коробки передач. Как мы видим из результатов ниже, первые два такта различаются. Это связано с особенностями обработки логики, описанными выше.
difference = gear_state.value-gear_node.value;
print("Различие в логике переключения передач: "*string(difference[1:3]))
Вывод
В данной демонстрации мы проанализировали различия в логике узлов и состояний. На этом примере мы получили наглядное представление о том, как обрабатываются разные алгоритмы, реализованные при помощи блока Chart.