5.ROS2-Topics-Publisher-Subscriber

文章目录

Topics

双方都不知道谁发的数据和谁接的数据,所以topic可以没有publishser或者subscriber

数据只能单向传输

Anonumous 匿名

python

template_python_node.py

python 复制代码
#! /usr/bin/env python3
import rclpy
from rclpy.node import Node

class MyCustomNode(Node): # MODIFY NAME
    def __init__(self):
        super().__init__("node_name") # MODIFY NAME:node name

def main(args=None):
    rclpy.init(args=args)
    node = MyCustomNode() # MODIFY NAME
    rclpy.spin(node)
    rclpy.shutdown()

if __name__ == "__main__":
    main()

Publisher

robot_news_station.py

python 复制代码
#! /usr/bin/env python3
import rclpy
from rclpy.node import Node
from example_interfaces.msg import String

class RobotNewsStationNode(Node): # MODIFY NAME
    def __init__(self):
        super().__init__("robot_news_station") # MODIFY NAME:node name
        self.publisher_ = self.create_publisher(String, "robot_news", 10) # data type, topic name, queue size
        self.timer_ = self.create_timer(1.0, self.publish_news)
        self.get_logger().info("Robot News Station Node has been started.")

    def publish_news(self):
        msg = String()
        msg.data = "Breaking news: ROS 2 is awesome!"
        self.publisher_.publish(msg)

def main(args=None):
    rclpy.init(args=args)
    node = RobotNewsStationNode() # MODIFY NAME
    rclpy.spin(node)
    rclpy.shutdown()

if __name__ == "__main__":
    main()
  • 一旦有新的import,在package.xml中添加依赖示例接口
xml 复制代码
<depend>example_interfaces</denpend>
  • setup.py中添加
python 复制代码
entry_points={
	'console_scripts':[
		"py_node = my_py_pkg.my_first_node:main",
		"robot_news_station = my_py_pkg.robot_news_station:main" # 新增的
	],
},

重新构建packages并运行

bash 复制代码
cd ~/ros2_ws
colcon build --packages-select my_py_pkg
source install/setup.bash
ros2 run my_py_pkg robot_news_station

可以通过topic指令查看运行的节点

bash 复制代码
ros2 topic list
ros2 topic echo /robot_news

Subscriber

python 复制代码
#! /usr/bin/env python3
import rclpy
from rclpy.node import Node
from example_interfaces.msg import String

class SmartphoneNode(Node):
    def __init__(self):
        super().__init__("smartphone") 
        self.robot_name = "nyw"
        self.subscriber_ = self.create_subscription(String, "robot_news", self.callback_robot_news, 10) # 使用相同的 topic 名称和publisher
        self.get_logger().info(f"Smartphone node initialized. Subscribed to 'robot_news' topic.")
    def callback_robot_news(self, msg: String):
        self.get_logger().info(f"Received news: {msg.data}")

def main(args=None):
    rclpy.init(args=args)
    node = SmartphoneNode()
    rclpy.spin(node)
    rclpy.shutdown()

if __name__ == "__main__":
    main()

setup.py添加

python 复制代码
entry_points={
	'console_scripts':[
		"py_node = my_py_pkg.my_first_node:main",
		"robot_news_station = my_py_pkg.robot_news_station:main",
		"smartphone = my_py_pkg.smartphone:main"# 新增的
	],
},

重新构建packages并运行

bash 复制代码
cd ~/ros2_ws
colcon build --packages-select my_py_pkg
source install/setup.bash
ros2 run my_py_pkg 

C++

template_cpp_node.cpp

c 复制代码
#include "rclcpp/rclcpp.hpp"

class MyCustomNode : public rclcpp::Node // MODIFY NAME
{
public:
    MyCustomNode() : Node("node_name") // MODIFY NAME
    {
    }

private:
};

int main(int argc, char **argv)
{
    rclcpp::init(argc, argv);
    auto node = std::make_shared<MyCustomNode>(); // MODIFY NAME
    rclcpp::spin(node);
    rclcpp::shutdown();
    return 0;
}

Publisher

c++ 复制代码
#include "rclcpp/rclcpp.hpp"
#include "example_interfaces/msg/string.hpp"

using namespace std::chrono_literals;

class RobotNewsStationNode : public rclcpp::Node
{
public:
    RobotNewsStationNode() : Node("robot_news_station")
    {
        publisher_ = this->create_publisher<example_interfaces::msg::String>("robot_news", 10), robot_name_ = "c++"; // 10的意思是在缓冲区最多10条信息

        // this->create_wall_timer(std::chrono::seconds(1), timer_callback); 因为前面using namespace std::chrono_literals;所以可以直接使用1s
        timer_ = this->create_wall_timer(1s, std::bind(&RobotNewsStationNode::publishNews, this)); // 绑定成员函数作为回调
        RCLCPP_INFO(this->get_logger(), "Robot News Station Node has been started.");
    }

private:
    void publishNews()
    {
        auto msg = example_interfaces::msg::String();
        msg.data = std::string("Hi, this is ") + robot_name_ + std::string(" reporting the latest news!");
        publisher_->publish(msg);
    }

    std::string robot_name_;
    rclcpp::Publisher<example_interfaces::msg::String>::SharedPtr publisher_; // 下划线是为了说明这是一个attribute
    rclcpp::TimerBase::SharedPtr timer_;
};

int main(int argc, char **argv)
{
    rclcpp::init(argc, argv);
    auto node = std::make_shared<RobotNewsStationNode>();
    rclcpp::spin(node);
    rclcpp::shutdown();
    return 0;
}
  • 新的include,在package.xml中加入
xml 复制代码
<depend>example_interfaces</depend>
  • CMakeLists.txt加入
c 复制代码
find_package(example_interfaces REQUIRED)
  • 转换可执行文件
    CMakeLists.txt加入
c++ 复制代码
add_executable(robot_news_station src/robot_news_station.cpp)
ament_target_dependencies(robot_news_station rclcpp example_interfaces)

install(TARGETS
	cpp_node
	robot_news_station
	DESTINATION lib/${PROJECT_NAME}
)

效果如下:

  • 构建package和运行
c++ 复制代码
cd ~/ros2_ws
colcon build --packages-select my_cpp_pkg
source install/setup.bash
ros2 run my_cpp_pkg robot_news_station

可以通过topic指令查看运行的节点

bash 复制代码
ros2 topic list
ros2 topic echo /robot_news_station

也可以启动python的smartphone来接受数据

相关推荐
jr-create(•̀⌄•́)1 小时前
LeakyRelu链式法则
开发语言·python·深度学习
vx_biyesheji00012 小时前
计算机毕业设计:Python股价预测与可视化系统 Flask框架 数据分析 可视化 机器学习 随机森林 大数据(建议收藏)✅
python·机器学习·信息可视化·数据分析·flask·课程设计
lulu12165440788 小时前
Claude Code项目大了响应慢怎么办?Subagents、Agent Teams、Git Worktree、工作流编排四种方案深度解析
java·人工智能·python·ai编程
Ares-Wang8 小时前
Flask》》 Flask-Bcrypt 哈希加密
后端·python·flask
kongba0079 小时前
项目打包 Python Flask 项目发布与打包专家 提示词V1.0
开发语言·python·flask
belldeep9 小时前
介绍 遗传算法 与 TSP问题
python·遗传算法·ga·tsp问题
解救女汉子9 小时前
SQL触发器如何获取触发源应用名_利用APP_NAME函数追踪
jvm·数据库·python
思绪无限10 小时前
YOLOv5至YOLOv12升级:血细胞检测系统的设计与实现(完整代码+界面+数据集项目)
人工智能·python·深度学习·目标检测·计算机视觉·yolov12·血细胞检测
skywalk816312 小时前
发现Kotti项目的python包Beaker 存在安全漏洞
开发语言·网络·python·安全