2.1.2 ROS通信编程

 话题编程
 服务编程
 动作编程

话题编程流程

 创建发布者
  初始化ROS节点;
  向ROS Master注册节点信息,包括发布的话题名和话题中的消息类型;
  按照一定的频率循环发布消息
  实践:在src中创建一个talker.cpp的文件

/**
 *该例程将发布chatter话题,消息类型为String
 */

#include <sstream>
#include "ros/ros.h"
#include "std_msgs/String.h"

int main(int argc,char **argv)
{
    //ROS节点初始化
    ros::init(argc,argv,"talker");

    //创建节点句柄(用来管理发布者订阅者等资源)
    ros::NodeHandle n;
    
    //创建一个Publisher,发布名为chatter的topic,消息类型为std_msgs::String
    ros::Publisher chatter_pub = n.advertise<std_msgs::String >("chatter",1000);
                        //发布话题的信息类型 话题名   缓冲区长度

    //设置循环频率      10HZ ->100ms(循环周期
    ros::Rate loop_rate(10);

    while (ros::ok())
    {
        //初始化std_msg::String类型的消息
        std_msgs::String msg;
        std::stringstream ss;
        ss<< "hello world"<< count;
        msg.data = ss.str();

        //发布消息
        ROS_INFO("%s",msg.data.c_str());
        chatter_pub.publish(msg);

        //循环等待回调函数
        ros::spinOnce();

        //按照循环频率延时
        loop_rate.sleep();
        ++count;            
    }
    return 0;
}

 创建订阅者
  初始化ROS节点;
  订阅需要的话题
  循环等待话题消息,接收到消息后进入回调函数;
  在回调函数中完成函数处理。
  实践:在src中创建一个listen.cpp的文件。

/**
 *该例程将订阅chatter话题,消息类型为String
 */

#include <ros/ros.h>
#include "std_msgs/String.h"

void chatterCallback(const std_msgs::String::ConstPtr& msg)
{
    //将接收到的消息打印出来
    ROS_INFO("I heard: [%s]",msg->data.c_str());
}
int main(int argc,char **argv)
{
    //初始化ROS节点
    ros::init(argc,argv,"listener");

    //创建句柄
    ros::NodeHandle n;

    //订阅chatter话题
    ros::Subscriber sub = n.subscribe("chatter",1000,chatterCallback);

    //循环等待话题
    ros::spin();

}

 添加编译选项
  设置需要编译的代码和生成的可执行文件
    由于CMakeLists是关于编译的,所以需要修改该文件

add_executable(talker src/talker.cpp) #生成可执行文件talker
target_link_libraries(talker ${catkin_LIBRARIES})                        #许多可执行文件是依赖第三方的库的,因此需要target_link_libraries(...),由于没有第三方的库,所以链接默认的库
add_executable(listener src/listener.cpp) 
target_link_libraries(listener ${catkin_LIBRARIES}) 

  设置链接库
  设置依赖
 运行可执行程序

自定义话题消息

1.定义msg文件
2.在package.xml中添加功能包依赖
3.在CMakeList.txt中添加编译选项
实践:一般自定义的话题在功能包的文件夹下面:


新建msg文件夹

创建Person.msg

在package.xml中添加功能包

在CMakeLists.txt添加编译选项

在CMakeLists.txt添加依赖

在CMakeLists.txt添加文件信息

查看消息是否编译成功

服务编程流程

 1.创建服务器
 2.创建客户端
 3.添加编译选项
 4.运行可执行程序
实践:与自定义话题步骤相似。由于请求/应答数据在ros中没有,所以应当自定义服务请求与应答。
 定义srv文件;


在~/catkin_ws/src/learning_communication/srv中添加.srv文件

定义srv文件

在package.xml中添加功能依赖

在CMakeList.txt中添加编译选项

在CMakeList.txt中添加编译选项

在CMakeList.txt中添加编译选项

编译检查是否成功
1.创建服务端
如何实现一个服务器:

1.初始化ROS节点;
2.创建Server实例;
3.循环等待服务请求,进入回调函数;
4.在回调函数中完成服务功能的处理,并反馈应答数据。
实践:
编写server.cpp:

/**
 *AddTwoInts Server
 */

#include "ros/ros.h"
#include "learning_communication/AddTwoInts.h"

//service回调函数,输入参数req,输出参数res
bool add(learning_communication::AddTwoInts::Request &req,
     learning_communication::AddTwoInts::Response &res)
{
    //将输入参数中的请求数据相加,结果放到应答变量中
    res.sum = req.a + req.b;
    ROS_INFO("request: x =%ld,y=%ld",(long int)req.a,(long int)req.b);
    ROS_INFO("sending back response:{%ld}",(long int)res.sum);

    return true;
}




int main(int argc,char **argv)
{

    //ROS节点初始化
    ros::init(argc,argv,"add_two_ints_server");
    
    //创建节点句柄
    ros::NodeHandle n;

    //创建一个名叫add_two_ints的server,注册回调函数为add()
    ros::ServiceServer service = n.advertiseService("add_two_ints",add);
    
    //循环等待回调函数
    ROS_INFO("Ready to add two ints:");
    ros::spin();

    return 0;

}

2.创建客户端
如何实现一个客户端

 初始化ROS节点;
 创建一个Client实例;
 发布服务请求数据;
 等待Server处理之后的应答结果;
实践:

#include <cstdlib>
#include "ros/ros.h"
#include "learning_communication/AddTwoInts.h"

int main(int argc,char **argv)
{
    //初始化节点
    ros::init(argc,argv,"add_two_ints_client");
    
    //从终端命令行获取两个加数
    if(argc != 3)
    {
        ROS_INFO("usage:add_two_ints_client X Y");
        return 1;
    }
    
    //创建节点句柄
    ros::NodeHandle n;
    
    //创建一个client,请求add_two_ints service,service消息类型是learning_communication::AddTwoInts
    ros::ServiceClient client = n.serviceClient<learning_communication::AddTwoInts>("add_two_ints");

    //创建一个learning_communication::AddTwoInts类型的srv消息
    learning_communication::AddTwoInts srv;
    srv.request.a = atoll(argv[1]);
    srv.request.b = atoll(argv[2]);

    //发布service请求,等待加法运算的应答结果
    if(client.call(srv))
    {
        ROS_INFO("Sum:%ld",(long int)srv.response.sum);
    }
    else
    {
        ROS_ERROR("Failed to call service add_two_ints");
        return 1;
    }

    return 0;

}
3.添加编译选项
如何编译代码

 设置需要编译的代码和生成的可执行文件;
 设置链接库;
 设置依赖。
实践:


修改CMakeList.txt

编译成功

生成可执行文件
4.运行可执行程序
运行可执行程序

动作编程

什么是动作:

 一种问答通信机制;
 带有连续反馈;
 可以在任务过程中终止运行
 基于ROS的消息机制实现

Action的接口

 goal:发布任务目标;
 cancel:请求取消任务;
 status:通知客户端当前的状态;
 feesback:周期反馈任务运行的监控数据
 result:向客户端发送任务的执行结果,只发布一次。

如何自定义动作消息

 定义action文件;


定义action文件

 在package.xml中添加功能包依赖


添加功能包依赖

 在CMakeList.txt添加编译选项
添加编译选项

添加编译选项
如何实现一个动作的服务器

 初始化ROS节点;
 创建动作服务器实例;
 启动服务器,等待动作请求;
 在回调函数中完成动作服务器功能的处理,并反馈进度信息;
 动作完成,发送结束信息。
实践:

#include <ros/ros.h>
#include <actionlib/server/simple_action_server.h>
#include "learning_communication/DoDishesAction.h"

typedef actionlib::SimpleActionServer<learning_communication::DoDishesAction> Server;

//收到action的goal后调用该回调函数
void execute(const learning_communication::DoDishesGoalConstPtr& goal,Server* as)
{
    ros::Rate r(1);
    learning_communication::DoDishesFeedback feedback;

    ROS_INFO("Dishwasher %d is working.",goal->dishwasher_id);
    
    //假设洗盘子的进度,并且按照1Hz的频率发布进度feedback
    for(int i=1;i<=10;i++)
    {
        feedback.percent_complete = i*10;
        as->publishFeedback(feedback);
        r.sleep();
    }
    
    //当action完成后,向客户端返回结果
    ROS_INFO("Dishwasher %d finish working.",goal->dishwasher_id);
    as->setSucceeded();
}




int main(int argc,char** argv)
{

    //初始化ROS节点
    ros::init(argc,argv,"do_dishes_server");
    
    //创建句柄
    ros::NodeHandle n;

    //定义一个服务器
    Server server(n,"do_dishes",boost::bind(&execute,_1,&server),false);

    //服务器开始运行
    server.start();

    ros::spin();

    return 0;
}
如何实现一个动作客户端

 初始化ROS节点;
 创建动作客户端实例;
 连接动作服务端;
 发送动作目标;
 根据不同类型的服务端反馈处理回调函数。

#include <actionlib/client/simple_action_client.h>
#include "learning_communication/DoDishesAction.h"

typedef actionlib::SimpleActionClient<learning_communication::DoDishesAction> Client;

//当action完成后会调用改回调函数一次
void doneCb(const actionlib::SimpleClientGoalState& state,
    const learning_communication::DoDishesResultConstPtr& result)
{
    ROS_INFO("Yay! The dishes are now clean");
    ros::shutdown();
}

//当action激活后会调用该回调函数一次
void activeCb()
{
    ROS_INFO("Goal just went active");
}

//收到feedback后调用改回调函数
void feedbackCb(const learning_communication::DoDishesFeedbackConstPtr& feedback)
{
    ROS_INFO(" percent_complete : %f",feedback->percent_complete);
}



int main(int argc,char** argv)
{
    //初始化ROS
    ros::init(argc,argv,"do_dishes_client");
    
    //定义一个客户端
    Client client("do_dishes",true);

    //等待服务器端
    ROS_INFO("Waiting for action server to start.");
    client.waitForServer();
    ROS_INFO("Action server started,sending goal.");

    //创建一个action的goal
    learning_communication::DoDishesGoal goal;
    goal.dishwasher_id = 1;
    
    //发送action的goal给服务器端,并且设置回调函数
    client.sendGoal(goal,&doneCb,&activeCb,&feedbackCb);

    ros::spin();
    
    return 0;
}
如何编译代码

 设置需要编译的代码和生成的可执行文件;
 设置链接库
 设置依赖
实践:

add_executable(DoDishes_client src/DoDishes_client.cpp) 
target_link_libraries(DoDishes_client ${catkin_LIBRARIES})
add_dependencies(DoDishes_client ${${PROJECT_NAME}_EXPORTED_TARGETS})  

add_executable(DoDishes_server src/DoDishes_server.cpp) 
target_link_libraries(DoDishes_server ${catkin_LIBRARIES})
add_dependencies(DoDishes_server ${${PROJECT_NAME}_EXPORTED_TARGETS})  
如何运行可执行文件
编译并执行

注意:在编译的时候报了这样的错误

zyt@zyt:~$ rosrun learning_communication DoDishes_server 
terminate called after throwing an instance of 'ros::InvalidNameException'
  what():  Character [ ] at element [2] is not valid in Graph Resource Name [do dishes server].  Valid characters are a-z, A-Z, 0-9, / and _.
已放弃 (核心已转储)

原因是在DoDishes_server.cpp中ros::init(argc,argv,"do_dishes_server");这一句的"_"写成了空格,空格是不能用来命名的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容

  • 需要一个人有强大的上进心、克制力、自律。克制自己的欲望,希望自己能做到"精进"。 话题编程 创建发布者 创建订阅者...
    徐凯_xp阅读 1,240评论 0 1
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,085评论 1 32
  • feisky云计算、虚拟化与Linux技术笔记posts - 1014, comments - 298, trac...
    不排版阅读 3,813评论 0 5
  • 大家好,接下来一段时间,在学习机器学习算法的同时,我也将逐步学习机器人操作系统ROS (Robot Operati...
    keepStriving阅读 38,141评论 2 35
  • 概要 64学时 3.5学分 章节安排 电子商务网站概况 HTML5+CSS3 JavaScript Node 电子...
    阿啊阿吖丁阅读 9,095评论 0 3