ROS2从入门到精通1-1:详解ROS2话题通信机制与自定义消息

        publisher_->publish(msg);
    }

private:
    rclcpp::Publisher<geometry_msgs::msg::Twist>::SharedPtr publisher_;

};

int main(int argc, char * argv[])
{
rclcpp::init(argc, argv);

auto node = std::make\_shared<PublisherNode>();

rclcpp::Rate loop\_rate(10);
while(rclcpp::ok()) {
    auto msg = geometry_msgs::msg::Twist();
    msg.linear.x = 0.5;
    msg.angular.z = 0.2;
    node->publish(msg);
    RCLCPP\_INFO(node->get\_logger(), "Publishing: x: %.2f, z: %.2f", msg.linear.x, msg.angular.z);
    loop_rate.sleep();
}   

rclcpp::shutdown();                                
return 0;

}

* **订阅者**(订阅的话题存在消息即触发回调函数)

 

class SubscriberNode : public rclcpp::Node
{
public:
SubscriberNode() : Node(“lab_topic_sub”)
{
subscriber_ = create_subscription<geometry_msgs::msg::Twist>(
“/turtle1/cmd_vel”, 10, std::bind(&SubscriberNode::OnPoseCallback, this, _1));
}

private:
    void OnPoseCallback(const geometry_msgs::msg::Twist & msg) const
    {
        RCLCPP\_INFO(get\_logger(), "Publishing: x: %.2f, z: %.2f", msg.linear.x, msg.angular.z);
    }
    
    rclcpp::Subscription<geometry_msgs::msg::Twist>::SharedPtr subscriber_; 

};

int main(int argc, char * argv[])
{
rclcpp::init(argc, argv);
rclcpp::spin(std::make_shared());
rclcpp::shutdown();

return 0;

}



话题通信的效果如下所示:


![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/1ef946f944614f30b21d9a189f62cf14.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBATXIuV2ludGVyYA==,size_40,color_FFFFFF,t_70,g_se,x_16#pic_center)


计算图可视化为:


![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/08bfd0969f15497894d4d6e8098a554c.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBATXIuV2ludGVyYA==,size_40,color_FFFFFF,t_70,g_se,x_16,y_82#pic_center)


## 3 话题模型实现(Python)



> 
> 实验目标:发布者发布控制消息到`/turtle1/cmd_vel`,控制乌龟其做圆周运动;订阅者订阅`/turtle1/cmd_vel`,在终端显示乌龟实时的位置坐标。
> 
> 
> 


* **发布者**

class PublisherNode(Node):
def __init__(self, name):
super().init(name)
self.publisher_ = self.create_publisher(Twist, ‘/turtle1/cmd_vel’, 10)

def publish(self, msg: Twist):
    self.publisher_.publish(msg)
* **订阅者**

class SubscriberNode(Node):
def __init__(self, name):
super().init(name)
self.subscirber_ = self.create_subscription(
Twist, ‘/turtle1/cmd_vel’, self.OnPoseCallback, 10
)

def OnPoseCallback(self, msg):
    self.get_logger().info(f"Publishing: x: {msg.linear.x:.2f}, z: {msg.angular.z:.2f}")

def main(args=None):
rclpy.init(args=args)
node = SubscriberNode(“lab_topic_sub”)
rclpy.spin(node)
node.destroy_node()
rclpy.shutdown()



话题通信的效果如下所示:


![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/d57242f314c74759aa4cc07376acce9e.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBATXIuV2ludGVyYA==,size_40,color_FFFFFF,t_70,g_se,x_16,y_82#pic_center)


## 4 自定义消息


ROS2系统通过`std_msgs`封装了一些常用的原生数据类型,比如`String`、`Int32`、`Int64`等,对于一些复杂数据应用场景,往往需要在`std_msgs`或其他消息库的基础上继续封装更高级的数据类型


自定义消息的通用流程如下:



> 
> * 功能包下新建`msg`文件夹,在其中添加自定义消息`xxx.msg`
> * 功能包`package.xml`中添加编译依赖与执行依赖
> ```
> <buildtool\_depend>rosidl_default_generators</buildtool\_depend>
> <exec\_depend>rosidl_default_runtime</exec\_depend>
> <member\_of\_group>rosidl_interface_packages</member\_of\_group>
> 
> ```
> * 功能包`CMakeLists.txt`中添加编译消息相关依赖
> ```
> find_package(rosidl_default_generators REQUIRED)
> rosidl_generate_interfaces(${PROJECT_NAME}
> 	"xxx.msg"
> 	DEPENDENCIES xxx_msgs
> )
> 
> ament_export_dependencies(rosidl_default_runtime)
> 
> ```
> * 编译自定义消息,在`install/<pkg_name>/include`中生成由`xxx.msg`编译的C++可识别的`xxx.hpp`头文件
> * 引入`xxx.hpp`即可调用自定义消息
> 
> 
> 


下面给出一个实例:


添加如下自定义消息,并按上面步骤配置依赖



Person.msg

string name
string gender
uint8 age

geometry_msgs/Point position
float64 x
float64 y
float64 z


定义一个发布者、一个订阅者测试自定义消息


* **发布者**

// 初始化名为personPub的ROS节点, 该节点应在CMakeLists.txt中被构建为可执行文件
ros::init(argc, argv, “personPub”);
// 创建节点句柄
ros::NodeHandle pubNode;
// 创建发布者, 该发布者属于pubNode节点, 发布话题为"/person/info",
// 消息类型为"msgtest::Person", 发布队列长度为10
ros::Publisher pub = pubNode.advertise<msg_lab::Person>(“/person/info”, 10);
// 设置发布频率
ros::Rate loopRate(10);
while(ros::ok())
{
// 设置消息
msg_lab::Person msg;
msg.name = “winter”;
msg.gender = “man”;
msg.age = 20;
msg.position.x = 10;
msg.position.y = 20;
msg.position.z = 30;
// 发布消息
pub.publish(msg);
ROS_INFO(“Publish Person Info[name: %s gender: %s age: %d pos: x-%.2f y-%.2f z-%.2f]”,
msg.name.c_str(), msg.gender.c_str(), msg.age, msg.position.x, msg.position.y, msg.position.z);
// 按循环频率延时
loopRate.sleep();
}

* **订阅者**

void personInfoCallBack(const msg_lab::Person::ConstPtr &msg)
{
ROS_INFO(“Subscribe Person Info[name: %s gender: %s age: %d pos: x-%.2f y-%.2f z-%.2f]”,
msg->name.c_str(), msg->gender.c_str(), msg->age, msg->position.x, msg->position.y, msg->position.z);
}

int main(int argc, char** argv)
{
// 初始化名为personSub的ROS节点, 该节点应在CMakeLists.txt中被构建为可执行文件
ros::init(argc, argv, “personSub”);
// 创建节点句柄
ros::NodeHandle subNode;
// 创建订阅者, 该订阅者属于subNode节点, 订阅话题为"/person/info",
// 订阅队列长度为10, 收到订阅消息后出发回调函数personInfoCallBack
ros::Subscriber sub = subNode.subscribe(“/person/info”, 10, personInfoCallBack);
// 循环等待回调函数
ros::spin();
return 0;
}



实测效果如下:


![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/4349504d0bdd4d099497a11b494a83be.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBATXIuV2ludGVyYA==,size_30,color_FFFFFF,t_70,g_se,x_16,y_82#pic_center)


ROS2编程基础课程文档 ROS 2(机器人操作系统2)是用于机器人应用的开源开发套件。ROS 2之目的是为各行各业的开发人员提供标准的软件平台,从研究和原型设计再到部署和生产。 ROS 2建立在ROS 1的成功基础之上,ROS 1目前已在世界各地的无数机器人应用中得到应用。 特色 缩短上市时间 ROS 2提供了开发应用程序所需的机器人工具,库和功能,可以将时间花在对业务非常重要的工作上。因为它 是开源的,所以可以灵活地决定在何处以及如何使用ROS 2,以及根据实际的需求自由定制,使用ROS 2 可以大幅度提升产品和算法研发速度! 专为生产而设计 凭借在建立ROS 1作为机器人研发的事实上的全球标准方面的十年经验,ROS 2从一开始就被建立在工业级 基础上并可用于生产,包括高可靠性和安全关键系统。 ROS 2的设计选择、开发实践和项目管理基于行业利 益相关者的要求。 多平台支持 ROS 2在Linux,Windows和macOS上得到支持和测试,允许无缝开发和部署机器人自动化,后端管理和 用户界面。分层支持模型允许端口到新平台,例如实时和嵌入式操作系统,以便在获得兴趣和投资时引入和推 广。 丰富的应用领域 与之前的ROS 1一样,ROS 2可用于各种机器人应用,从室内到室外、从家庭到汽车、水下到太空、从消费 到工业。 没有供应商锁定 ROS 2建立在一个抽象层上,使机器人库和应用程序与通信技术隔离开来。抽象底层是通信代码的多种实现, 包括开源和专有解决方案。在抽象顶层,核心库和用户应用程序是可移植的。 建立在开放标准之上 ROS 2中的默认通信方法使用IDL、DDS和DDS-I RTPS等行业标准,这些标准已广泛应用于从工厂到航空 航天的各种工业应用中。 开源许可证 ROS 2代码在Apache 2.0许可下获得许可,在3条款(或“新”)BSD许可下使用移植的ROS 1代码。这两个 许可证允许允许使用软件,而不会影响用户的知识产权。 全球社区 超过10年的ROS项目通过发展一个由数十万开发人员和用户组成的全球社区,为机器人技术创建了一个庞大 的生态系统,他们为这些软件做出贡献并进行了改进。 ROS 2由该社区开发并为该社区开发,他们将成为未 来的管理者。 行业支持 正如ROS 2技术指导委员会成员所证明的那样,对ROS 2的行业支持很强。除了开发顶级产品外,来自世界 各地的大大小小公司都在投入资源为ROS 2做出开源贡献。 与ROS1的互操作性 ROS 2包括到ROS 1的桥接器,处理两个系统之间的双向通信。如果有一个现有的ROS 1应用程序, 可 以通过桥接器开始尝试使用ROS 2,并根据要求和可用资源逐步移植应用程序。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值