惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

Stack Overflow Blog
Stack Overflow Blog
WordPress大学
WordPress大学
罗磊的独立博客
S
Secure Thoughts
Schneier on Security
Schneier on Security
博客园 - Franky
www.infosecurity-magazine.com
www.infosecurity-magazine.com
Exploit-DB.com RSS Feed
Exploit-DB.com RSS Feed
爱范儿
爱范儿
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Hacker News: Ask HN
Hacker News: Ask HN
PCI Perspectives
PCI Perspectives
Google DeepMind News
Google DeepMind News
S
Security Affairs
SecWiki News
SecWiki News
博客园 - 聂微东
Security Archives - TechRepublic
Security Archives - TechRepublic
Google Online Security Blog
Google Online Security Blog
H
Heimdal Security Blog
S
Security @ Cisco Blogs
Engineering at Meta
Engineering at Meta
C
CXSECURITY Database RSS Feed - CXSecurity.com
Cloudbric
Cloudbric
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
V
Visual Studio Blog
P
Proofpoint News Feed
Project Zero
Project Zero
T
Threat Research - Cisco Blogs
Webroot Blog
Webroot Blog
Blog — PlanetScale
Blog — PlanetScale
K
KPMG report finds enterprise disconnect between AI and its ROI | CIO
W
WeLiveSecurity
Last Week in AI
Last Week in AI
月光博客
月光博客
Microsoft Azure Blog
Microsoft Azure Blog
M
MIT News - Artificial intelligence
有赞技术团队
有赞技术团队
S
Securelist
GbyAI
GbyAI
Application and Cybersecurity Blog
Application and Cybersecurity Blog
C
CERT Recently Published Vulnerability Notes
Recent Commits to openclaw:main
Recent Commits to openclaw:main
Cyberwarzone
Cyberwarzone
B
Blog RSS Feed
P
Palo Alto Networks Blog
H
Hacker News: Front Page
D
Docker
雷峰网
雷峰网
Latest news
Latest news
Microsoft Security Blog
Microsoft Security Blog

Все публикации подряд на Хабре

Ловим музу за клавиатуру: как айтишнику стать автором Что умеет Midjourney в 2026? Мой немного грустный разбор этого шикарного инструмента Никто не любит писать тесты, но ИИ может исправить это IPv8 выглядит как мечта. Поэтому почти наверняка не взлетит Производители вернули в продажу материнки с DDR3. Что происходит? Управление агентом с телефона через Telegram теперь в KodaCode От координации к лидерству: как меняется роль руководителя разработки Я сделала родителям бизнес вместо пенсии: зарабатываем 70 тысяч, мама не даёт продать В три раза быстрее приемка товара и оптимизация трудозатрат на 73%: как «РСТ-Инвент» помог Gulliver Group ИИ-шечный мир победил? О влиянии искусственного интеллекта на игропром Кремль снижает давление на Телеграмм пока Европа строит интернет по паспорту Как CEO, CTO и CIO за 8 часов собрали ИИ-директора, который умеет держать позицию под давлением Как (не) потерять домен за выходные Вместо 8 разных VPS: как я организовал практику студентам на одном сервере Почему твой Open Source проект не замечают? R&D: искусство управления неопределенностью в разработке AI-дефляция: вакансий для разработчиков больше, а рост зарплат — худший за 15 лет Мы отдали управление роботами OpenClaw. Что из этого вышло Галактический ID: система идентификации для всех форм разумной жизни Шесть основ бизнес-анализа: начинаем с вопроса «Кто в игре?» Код-ревью, в котором дело не в коде Данные переехали. Команда — нет Системной подход к сдаче OSWE в 2025 Почему комната управления реактором покрашена в цвет морской пены 4 YAML-файла вместо PySpark: как аналитикам строить пайплайны без разработчиков LLM-агент для поиска свободных доменов: автоматизируем подбор Когда, зачем и как правильно начинать новую сессию в Claude Code? Как я заставил нейросеть писать макросы для FreeCAD Анатомия ИИ‑агента для подбора персонала. От тысячи резюме к топ‑10 за минуты Опыт разработчика как экономика внимания Автономность как точка невозврата: кто будет субъектом в цифровом будущем Обучение ИИ в «диких» условиях: как рутинные действия превращаются в датасеты Как измерить LLM для задач кибербеза: обзор открытых бенчмарков Где хранить код? Сравнение GitHub, GitLab и Bitbucket Математика объясняет, почему нормальное распределение встречается повсюду Почему ваш FinOps не работает: 12 тезисов от практиков Как подписать проектную документацию УКЭП с использованием бесплатных лицензий Pilot Адаптивное администрирование Sigla Vision Я грузил уран в бочки, а потом 20 лет строил ИТ в атомной отрасли Чем позвонить с Эвереста? История и обзор спутниковой связи. Часть 2 Как языковая модель помогает контролировать качество инструктажей по охране труда в металлургии Как не передать на desktop свой IP в РКН Анатомия SAP Privileges: как устроено управление правами в macOS MoneyDev: Сказка про три главных слова Обновлённый токенизатор видео K-VAE 2.0 от Сбера Как сделать диспетчеризацию дома на 1284 квартиры почти бесплатно Как мы разогнали железную дорогу Мы дали агентам рутину. Теперь надо решить — что делать с освободившимся временем Токсичный контент, промпт-хакинг и защита ИИ — всё о Guardrails для LLM Умный город начинается с точного взгляда: как «Фалькон Тех» меняет пространство к лучшему Навайбкодил приложение для анализа графов Почему Дюну так интересно читать? Упрощаем работу с рутиной или как стать Гендальфом Белым Деконструкция Go: CPU, RAM и что там происходит. Go Assembler база. Часть 1.1 Какие профессии исчезнут из-за ИИ, а какие появятся? И что с этим делать Как мы построили IT-отдел, где хочется расти: архитектурные встречи, прозрачные метрики и книжные подарки Rufler: Делаем из Claude Code автономный рой через один YAML-конфиг Sing-box и белый список приложений Как построить надёжный обмен сообщениями в микросервисах: лучшие практики для enterprise OpenAI строит MLM-пирамиду, а McKinsey и Accenture помогают ей в этом Дом, который не построил Фишер (Часть 2) «Сверхзвуковой математик» против «Вдумчивого логиста»: битва алгоритмов 3D-упаковки Мультимодальные модели – грубый и дорогой инструмент Разговоры ничего не стоят. Код тоже Проверки физических лиц: с кого начнет ФНС Топ-10 бесплатных нейросетей для создания видео в 2026 году Первые слои кода: как наши решения сегодня определяют архитектуру ИИ на десятилетия Разработка нового статического анализатора: PVS-Studio JavaScript Поиск уязвимостей ПО: базовый минимум или роскошный максимум Почему оценка персонала не работает как инструмент управления Как мы разработали ИИ-ассистента и сократили рутину продуктовой команды на 50% Как я ушел из найма, нажарил косточек и продал на маркетплейсах на 168 млн в год Когда 1С:ERP уже внедрена, а нормального производственного плана всё ещё нет Как я сделал Claude мультимодальным, подключив к нему Qwen Omni Как приглашение на вакансию мечты превращается в атаку Infrastructure as Code: философия и лучшие практики IaC Тестируем Yandex Code Assistant на задаче, в которой нужно хранить секреты nxs-universal-chart v3.0: новое поколение универсального Helm-чарта Callback Injection: Техника, которая отправила Microsoft Defender в глухой нокаут «Все идеи на стол»: митап как способ вывести проект из тупика Сегодня я узнал нечто новое о GPU благодаря багу в своей игре Как заставить LLM ̶ ̶г̶а̶л̶л̶ю̶ ̶ эволюционировать Карта событий как фундамент аналитики: практический кейс для E-commerce Что выбрать для AI: x86, ARM или RISC-V? Дайджест железа за март Роль соматических мутаций в развитии аутоиммунных заболеваний: путь к избирательной терапии Mythos от Anthropic — тревожный сигнал для всех, а не только для банков Guardrails для LLM на Java: как приручить промпт‑инъекции и токсичные ответы Green-VLA: как мы собрали VLA-модель для реального антропоморфного робота и не потеряли обобщение Финансовая гонка вооружений: почему умные люди добровольно в ней участвуют Эра ИИ-агентов наступила: выбираем лучшего цифрового сотрудника # Практический опыт внедрения WinCC Redundancy на производственном предприятии Сделал MVP за 3 дня, а потом неделю прикручивал оплату. Оно того стоило? Физика против Маска: почему Starship V3 может оказаться ещё одной катастрофой Нефть Венесуэлы: крупнейшие запасы в мире, но не крупнейшая нефтяная держава JPA 4. Переосмысление Hibernate Почему зеркальная фотокамера Nikon D5 десятилетней давности идеально подошла для миссии «Артемида-2» Проект «Уровень-Спутник» или как мы сделали платформу для гидрологов «Замедлиться, чтобы ускориться»: почему ИИ повышает цену ошибок в требованиях и архитектуре Как с нуля поднять трафик IT-компании на 1657% при бюджете 55 тыс. и выжить Pixel-perfect Downsampling — идеальная отрисовка 50 миллионов точек без потерь
Создаем клиентскую библиотеку ROS2. Элементы ноды
stanislav_mi · 2026-05-18 · via Все публикации подряд на Хабре

Время на прочтение7 мин

Охват и читатели195

Роадмэп

Теперь, когда в вашей клиентской библиотеке появились сообщения, можно перейти к созданию ROS2 ноды и связанных с ней элементов (издателей, сервисов и т.п.). На самом деле процесс добавления этих элементов мало отличается от создания обёртки для таймера, описанный в первой части, поскольку всё сводится к надстройке над функциями библиотеки rcl. Поэтому я не буду рассматривать их подробно, а остановлюсь только на отдельных особенностях.

Нода

Нода (node) так названа, потому что является узлом ROS2 графа, ребрами же являются топики - именованные каналы обмена сообщениями. На уровне rcl её функции сводятся к вызовам типа rcl_node_get_namercl_count_publishersrcl_action_get_client_names_and_types_by_node, т.е. к получению информации о графе и своей роли в нем. Если же переходить на уровень реализации клиентской библиотеки на целевом языке программирования, функция ноды расширяется: теперь она хранит в себе все элементы, связанные с обменом сообщениями, а также часы, таймеры, логгеры и всё что вы захотите в неё добавить. Причин такой трансформации я вижу две. Во-первых, большинство этих элементов требуют указатель на объект ноды при инициализации. Во-вторых, в функцию spin() проще передать один контейнер, чем каждый элемент по отдельности. В результате, конструктор ноды (в имплементации на Lua) приобретает следующую структуру.

function Node.__call (self, ...)
  local o = {}
  -- rcl нода
  o._node__object = rclbind.new_node(self.name, self.namespace)
  -- имя
  o._node__name = self.name
  -- добавляемые элементы
  o._clock__object = rclbind.new_clock()
  o._timer__list = {}
  o._publisher__list = {}
  o._subscription__list = {}
  o._client__list = {}
  o._service__list = {}
  o._action__list = {}
  o._guard__list = {}
  -- наследуем методы ноды
  return setmetatable(o, Node)
end

Издатели и подписчики

Издатель (publisher) и подписчик (subscriber) служат для широковещательной трансляции данных: для каждого топика число издателей и подписчиков может быть произвольным. При создании они требуют указывать связанную ноду, тип сообщения, имя топика и QoS канала связи.

rcl_publisher_options_t publisher_opt = rcl_publisher_get_default_options();
rcl_publisher_t publisher =  rcl_get_zero_initialized_publisher();
// инициализация издателя, аналогично для подписчика
rcl_ret_t ret = rcl_publisher_init(publisher, node, message_type, topic, &publisher_opt);

Публикация сообщения сводится к вызову функции rcl_publish(publisher, message). Что касается подписчика, нужно дополнительно связать callback-функцию с указателем на объект на уровне целевого языка программирования, поскольку в rcl такой функционал не предусмотрен.

Сервисы и клиенты

Как понятно из названия, логика работы стандартная: клиент посылает запрос, сервис его обрабатывает и возвращает ответ. Сервис может обрабатывать запросы многих клиентов, но последовательно; клиент работает с одним сервисом, но может послать несколько запросов, не дожидаясь ответа. Отсюда вытекают требования к реализации: сервис должен понимать, кому возвращать результат, а клиент должен знать, которому из запросов соответствует полученный ответ. Первая задача решается сохранением id клиента, вторая - id запроса. В обоих случаях для хранения данных между вызовами используются какие-то сущности (классы, контейнеры), которые должны освобождаться после передачи/получения ответа. Для обмена данными служат следующие функции rcl.

// сервис
  // получение запроса
  rmw_service_info_t header;  // сохранение информации о клиенте
  rcl_ret_t ret = rcl_take_request_with_info(srv, &header, request);
  // передача результата
  rcl_ret_t ret = rcl_send_response(srv, &header->request_id, response);

  // -------------------

  // клиент
  // передача запроса
  int64_t seq_num = 0;
  rcl_ret_t ret = rcl_send_request(cli, request, &seq_num);
  // получение результата
  rmw_service_info_t header;
  rcl_ret_t ret = rcl_take_response_with_info(cli, &header, response);
  // идентификатор находится в header.request_id.sequence_number

При работе с клиентом у пользователя есть выбор: остановить выполнение программы до получения ответа или продолжить работу. На уровне rcl нет синхронного или асинхронного вызова, логика работы с Wait Set будет одинаковая в обоих случаях, поэтому данный функционал реализуется с помощью целевого языка программирования.

Action сервисы и клиенты

Экшн сервисы отличаются от обычных продолжительностью действия: запрос клиента запускает процесс, за которым можно следить с помощью сообщений обратной связи. Этот процесс может завершиться успешно или не успешно, а также может быть прерван по инициативе клиента. Одно из ключевых особенностей реализации заключается в том, что нода должна иметь возможность запустить несколько action сервисов одновременно, т.е. они должны выполняться в параллельных потоках, в то время как весь прочий функционал ROS2 по умолчанию однопоточный.

После того как экшн клиент послал запрос серверу, логика его работы сводится к следующему циклу. Wait Set имеет отдельный метод для работы с экшн клиентом и сервисом. При завершении ожидания возвращаются флаги, которые указывают, какое из сообщений было получено: ответ сервиса регистрации задания (goal), ответ сервиса результата (result), ответ сервиса прерывания исполнения (cancel), сообщение обратной связи (feedback), сообщение о состоянии работы сервера (status). Исходя из этого запускается соответствующая логика обработки.

while rclbind.context_ok() do
  -- добавляем в Wait Set
  wait_set:clear()
  act_cli:add_to_waitset(wait_set)
  -- ждем ответа action сервиса
  wait_set:wait(-1)

  -- извлекаем тип сообщения
  local is_feedback, is_status, is_goal, is_cancel, is_result = act_cli:is_ready(wait_set)

  -- ответ сервера на задание
  if is_goal then
    local resp, fn, seq = act_cli:take_goal_response()
    if resp.accepted then
      -- посылаем запрос на получения результата
    else
      -- сервер отклонил наше задание
    end
  end

  -- если клиент прервал работу сервера
  if is_cancel then
    local resp, fn, seq = act_cli:take_cancel_response()
    -- здесь можно вызвать callback для обработки
  end

  -- сервер прислал результат
  if is_result then
    local resp, fn, seq = act_cli:take_result_response()
    -- вызваем callback для полученных данных
  end

  -- сообщение обратной связи
  if is_feedback then
    local resp, fn = act_cli:take_feedback()
    -- следим за ходом работы
  end

  -- статус сервера (задание принято, выполняется, отменено и т.д.)
  if is_status then
    local resp, fn = act_cli:take_status()
    -- следим за сервером
  end
end

В свою очередь экшн сервер выполняет следующий цикл обработки. При получении запроса на запуск работ выполняются необходимые проверки, после чего задание принимается либо отклоняется. Результат работ не возвращается автоматически, его должен запросить клиент через сервис. Обратная связь обычно публикуется в каждом цикле обработки, а статус сервера только при изменении.

while rclbind.context_ok() do
  -- добавляем в Wait Set
  wait_set:clear()
  act_srv:add_to_waitset(wait_set)

  -- ждем новый запрос или заданное время
  wait_set:wait(time_ns)

  -- проверка входящих сообщений
  local is_goal, is_cancel, is_result, is_expired = act_srv:is_ready(wait_set)

  -- новое задание
  if is_goal then
    local req, _, header = act_srv:take_goal_request()
    -- проверяем req.goal_id.uuid чтобы избежать повторных запусков
    -- если задание новое, создаем поток для выполнения
    -- посылаем ответ клиенту (принято/не принято)
  end

  -- прерывание процесса
  if is_cancel then
    local req, _, header = act_srv:take_cancel_request()
    -- завершаем требуемые задачи
    -- посылаем ответ клиенту (список завершенных задач)
  end

  -- запрос результата
  if is_result then
    local req, _, header = act_srv:take_result_request()
    -- сохраняем информацию о клиенте
    -- чтобы позже вернуть результат
  end

  if is_expired then
    local lst = act_srv:expire_goals(handle_num)
    -- очистить список задач
  end

  -- в процессе выполнения нужно передавать клиенту
  -- сообщения обратной связи и статус задачи
end

Executor и spin

Для удобства работы с Wait Set в клиентской библиотеке ROS2 вводится класс Executor. Он выполняет всю работу по регистрации событий, их ожиданию и последующей обработке. Данного класса в rcl нет, он реализуется на уровне целевого языка программирования.

Объект Executor хранит ссылки на одну или несколько ROS2 нод. При вызове метода spin (или его вариаций spin_oncespin_until_future_complete) запускается цикл, на каждой итерации которого из всех нод собираются связанные с ними элементы (издатели, таймеры, сервисы и т.д.), добавляются в Wait Set, и после разблокировки выполняются соответствующие callback функции. Данный алгоритм может быть вынесен в следующую функцию.

function _wait_for_ready_callbacks (executor, timeout_sec)
  -- подготовка элементов для Wait Set
  for _, node in ipairs(executor._nodes) do
    -- извлечение списка подписчиков, таймеров, сервисов, клиентов, защитников, событий

    -- отдельно для action сервисов/клиентов
    for _, act in ipairs(node._action__list) do
      local sub_no, guard_no, timer_no, cli_no, srv_no = act:get_num_entities()
      sub_cnt   = sub_cnt + sub_no
      timer_cnt = timer_cnt + timer_no
      cli_cnt   = cli_cnt + cli_no
      srv_cnt   = srv_cnt + srv_no
      guard_cnt = guard_cnt + guard_no
    end
  end

  -- инициализация
  executor._wait_set = rclbind.new_wait_set(
    sub_cnt,
    guard_cnt,
    timer_cnt,
    cli_cnt,
    srv_cnt,
    ev_cnt)

  -- добавление событий
  local wait_set = executor._wait_set
  wait_set:clear()

  for i = 1, #subscriptions do wait_set:add_subscription(subscriptions[i]) end
  for i = 1, #timers do wait_set:add_timer(timers[i]) end
  for i = 1, #clients do wait_set:add_client(clients[i]) end
  for i = 1, #services do wait_set:add_service(services[i]) end

  for i = 1, #actions do actions[i]:add_to_waitset(wait_set) end

  -- ожидание
  wait_set:wait(timeout_sec)
  if not rclbind.context_ok() then return end

  -- проверка событий
  subscriptions = wait_set:ready_subscriptions()
  timers = wait_set:ready_timers()
  clients = wait_set:ready_clients()
  services = wait_set:ready_services()

  -- обработка
  for _, act in ipairs(actions) do
    -- извлечение сообщения для action сервера/клиента
    -- возврат функции обработки
  end

  for i = 1, #subscriptions do
    -- возврат callback функции подписчика
  end

  for i = 1, #timers do
    -- возврат callback функции таймера
    -- вызов метода call таймера
  end

  for i = 1, #services do
    -- возврат функции сервиса
    -- передача результата клиенту
  end

  for i = 1, #clients do
    -- возврат callback функций клиентов
  end
end

Обычно данная функция заворачивается в корутину, которая для каждого объекта в цикле возвращает исполняемый handle. Это позволяет отследить промежуточные события, такие как нажатие пользователем Ctrl+C или прерывание текущего цикла обработки через вызов триггера объекта guard.

-- однократное исполнение
function Executor.spin_once (self, timeout_sec)
  -- создание корутины (если требуется)
  self._cb_iter = coroutine.create(_wait_for_ready_callbacks)
  -- вызов
  local ok, handle = coroutine.resume(self._cb_iter, self, timeout_sec)
  -- обработка
  if ok and handle then handle() end
  -- прочие операции
end

-- непрерывное исполнение
function Executor.spin (self)
  while rclbind.context_ok() do
    Executor.spin_once(self, -1)
  end
end

При инициализации контекста окружения ROS2 помимо всего прочего создается дефолтная версия объекта Executor. Именно она запускает цикл обработки событий, если пользователь не создает Executor в явном виде.

Заключение

Функционал, рассмотренный в этих 3х статьях, не покрывает все возможности ROS2. В частности, я не написал про сервис параметров и lifecycle ноды. Однако, данный материал может помочь с реализаций большинства функций при интеграции нового языка программирования в экосистему ROS2. А также послужить путеводителем для тех, кто захочет погрузиться в исходные коды этого фреймворка.