使用 C++ 实现ROS话题通信

需求:

编写发布订阅实现,要求发布方以10HZ(每秒10次)的频率发布文本消息,订阅方订阅消息并将消息内容打印输出。

分析:

在模型实现中,ROS master 不需要实现,而连接的建立也已经被封装了,需要关注的关键点有三个:

  1. 发布方
  2. 接收方
  3. 数据(此处为普通文本)

流程:

  1. 编写发布方实现;
  2. 编写订阅方实现;
  3. 编辑配置文件;
  4. 编译并执行。
发布方publisher.cpp
/*
    需求: 实现基本的话题通信,一方发布数据,一方接收数据,
         实现的关键点:
         1.发送方
         2.接收方
         3.数据(此处为普通文本)

         PS: 二者需要设置相同的话题


    消息发布方:
        循环发布信息:HelloWorld 后缀数字编号

    实现流程:
        1.包含头文件 
        2.初始化 ROS 节点:命名(唯一)
        3.实例化 ROS 句柄
        4.实例化 发布者 对象
        5.组织被发布的数据,并编写逻辑发布数据

*/
// 1.包含头文件 
#include "ros/ros.h"
#include "std_msgs/String.h" //普通文本类型的消息
#include <sstream>

int main(int argc, char  *argv[])
{   
    //设置编码,使终端可以正确输出除英文外的其他语言
    setlocale(LC_ALL,"");

    //2.初始化 ROS 节点:命名(唯一)
    // 参数1和参数2 后期为节点传值会使用
    // 参数3 是节点名称,是一个标识符,需要保证运行后,在 ROS 网络拓扑中唯一
    ros::init(argc,argv,"talker");
    //3.实例化 ROS 句柄
    ros::NodeHandle nh;//该类封装了 ROS 中的一些常用功能

    //4.实例化 发布者 对象
    //泛型: 发布的消息类型
    //参数1: 要发布到的话题
    //参数2: 队列中最大保存的消息数,超出此阀值时,先进的先销毁(时间早的先销毁)
    ros::Publisher pub = nh.advertise<std_msgs::String>("chatter",10);

    //5.组织被发布的数据,并编写逻辑发布数据
    //数据(动态组织)
    std_msgs::String msg;
    // msg.data = "你好啊!!!";
    std::string msg_front = "Hello 你好!"; //消息前缀
    int count = 0; //消息计数器

    //逻辑(一秒10次)
    ros::Rate r(1);

    //休眠3秒
    ros::Duration(3).sleep();
    //节点不死
    while (ros::ok())
    {
        //使用 stringstream 拼接字符串与编号
        std::stringstream ss;
        ss << msg_front << count;
        msg.data = ss.str();
        //发布消息
        pub.publish(msg);
        
        //加入调试,打印发送的消息
        ROS_INFO("发送的消息:%s",msg.data.c_str());

        //根据前面制定的发送频率自动休眠 休眠时间 = 1/频率;
        r.sleep();
        count++;//循环结束前,让 count 自增
        
        //暂无应用
        ros::spinOnce();
    }


    return 0;
}
订阅方subscriber.cpp
/*
    需求: 实现基本的话题通信,一方发布数据,一方接收数据,
         实现的关键点:
         1.发送方
         2.接收方
         3.数据(此处为普通文本)


    消息订阅方:
        订阅话题并打印接收到的消息

    实现流程:
        1.包含头文件 
        2.初始化 ROS 节点:命名(唯一)
        3.实例化 ROS 句柄
        4.实例化 订阅者 对象
        5.处理订阅的消息(回调函数)
        6.设置循环调用回调函数

*/
// 1.包含头文件 
#include "ros/ros.h"
#include "std_msgs/String.h"

void doMsg(const std_msgs::String::ConstPtr& msg_p){
    ROS_INFO("我听见:%s",msg_p->data.c_str());
    // ROS_INFO("我听见:%s",(*msg_p).data.c_str());
}
int main(int argc, char  *argv[])
{
    setlocale(LC_ALL,"");
    //2.初始化 ROS 节点:命名(唯一)
    ros::init(argc,argv,"listener");
    //3.实例化 ROS 句柄
    ros::NodeHandle nh;

    //4.实例化 订阅者 对象
    ros::Subscriber sub = nh.subscribe<std_msgs::String>("chatter",10,doMsg);
    //5.处理订阅的消息(回调函数)

    //6.设置循环调用回调函数
    // ros::Duration(1.0).sleep();
    ros::spin();//循环读取接收的数据,并调用回调函数处理

    return 0;
}
配置 CMakeLists.txt
add_executable(publisher src/publisher.cpp)
add_executable(subscriber src/subscriber.cpp)

target_link_libraries(publisher
  ${catkin_LIBRARIES}
)
target_link_libraries(subscriber
  ${catkin_LIBRARIES}
)
执行
roscore
# 另开一个终端
source ./devel/setup.bash
rosrun 包名 publisher
# 另开一个终端
source ./devel/setup.bash
rosrun 包名 subscriber

PS:vscode 中的 main 函数 声明 int main(int argc, char const *argv[]){},默认生成 argv 被 const 修饰,需要去除该修饰符

<think>好的,我需要回答用户关于在ROS使用C++进行编译的问题,特别是关于编译配置的教程或示例。首先,我应该回忆ROS中的编译系统,即catkin或colcon,因为用户提到的ROS2通常使用colcon。用户提到的引用[1]和[2]提到了CMakeLists.txt的修改和软件包创建,这些是关键点。 首先,用户需要创建一个C++节点,然后配置CMakeLists.txt来编译它。我需要分步骤说明如何创建软件包、编写节点、修改CMakeLists.txt以及编译。要确保使用正确的指令,比如colcon build,并提到依赖项的添加,比如ament_cmake和rclcpp。同时,需要提到package.xml中的依赖配置,这点在引用[2]中提到了依赖项设置,但ROS2使用的是package.xml而不是ROS1的package.xml,所以需要调整说明。 另外,要确保用户了解源设置和环境变量的配置,比如source install/setup.bash。可能还需要指出常见的错误,比如依赖未正确声明或CMakeLists.txt配置错误。最后,生成相关问题,帮助用户深入理解,比如如何添加多个节点或处理依赖关系。需要保持回答结构清晰,步骤详细,并引用用户提供的参考资料中的内容,比如引用[1]中的CMakeLists.txt修改和引用[2]中的软件包创建部分。</think># ROS C++ 节点编译配置教程 ## 一、环境准备 1. 安装ROS2开发工具链(推荐Humble或Foxy版本) 2. 创建工作空间(若尚未创建): ```bash mkdir -p ~/ros2_ws/src cd ~/ros2_ws ``` ## 二、创建软件包 1. 在`src`目录下创建新软件包: ```bash ros2 pkg create my_package --build-type ament_cmake --dependencies rclcpp ``` *参数说明*: - `--build-type`: 指定编译系统类型(C++使用ament_cmake) - `--dependencies`: 声明依赖的核心库[^2] ## 三、编写C++节点 1. 在软件包内创建`src`目录并添加节点文件(例:hello_world.cpp): ```cpp #include "rclcpp/rclcpp.hpp" class HelloNode : public rclcpp::Node { public: HelloNode() : Node("hello_node") { RCLCPP_INFO(this->get_logger(), "Hello ROS2 World!"); } }; int main(int argc, char ** argv) { rclcpp::init(argc, argv); auto node = std::make_shared<HelloNode>(); rclcpp::spin(node); rclcpp::shutdown(); return 0; } ``` ## 四、配置编译文件 1. 修改`CMakeLists.txt`(关键配置示例): ```cmake # 声明C++标准 set(CMAKE_CXX_STANDARD 17) # 添加可执行文件 add_executable(hello_node src/hello_world.cpp) # 链接依赖库 ament_target_dependencies(hello_node rclcpp) # 安装目标到指定目录 install(TARGETS hello_node DESTINATION lib/${PROJECT_NAME}) ``` *注意*:此配置对应引用[1]中的CMakeLists修改要求[^1] 2. 配置`package.xml`(自动生成后验证): ```xml <depend>rclcpp</depend> ``` ## 五、编译与运行 1. 返回工作空间根目录执行编译: ```bash colcon build --symlink-install ``` 2. 加载环境变量: ```bash source install/setup.bash ``` 3. 运行节点: ```bash ros2 run my_package hello_node ``` ## 六、常见问题排查 1. **依赖缺失错误**:检查`package.xml`和`CMakeLists.txt`中的依赖声明 2. **C++标准不匹配**:确认`set(CMAKE_CXX_STANDARD 17)`已配置 3. **节点未找到**:检查install()指令的安装路径设置
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值