提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档
对于发布者来说,需要实现在发布话题的同时,提供一个服务,这时会出现ros::spin()与ros::spinonce()的冲突。经验证,服务与订阅者的回调函数只有在持续回调的过程中才有用。 对于接收者需要实现订阅的话题消息满足一定条件后,请求一个服务,这同样出现ros::spin()与client.call(srv)的冲突。
提示:以下是本篇文章正文内容,下面案例可供参考
对于服务节点的冲突在查阅相关资料后一般有两种思路。(首先说明一下对于多回调函数的处理方式) 一个节点中含有多回调函数时,通常使用的方法是定义一个类,回调函数在这个类中读取。然后在ros::spin()一下即可。 这种方法请看链接: link. 另一种方法就是通过多线程的方式实现的。 在ros中多线程可以分为同步线程和异步线程
个人感觉其想要解决的问题是多回调函数的问题,其实现方法如下:
#include "ros/ros.h" #include "std_msgs/String.h" #include <boost/thread.hpp> class multiThreadListener { public: multiThreadListener() { sub = n.subscribe("chatter1", 1, &multiThreadListener::chatterCallback1,this); sub2 = n.subscribe("chatter2", 1, &multiThreadListener::chatterCallback2,this); } void chatterCallback1(const std_msgs::String::ConstPtr& msg); void chatterCallback2(const std_msgs::String::ConstPtr& msg); private: ros::NodeHandle n; ros::Subscriber sub; ros::Subscriber sub2; }; void multiThreadListener::chatterCallback1(const std_msgs::String::ConstPtr& msg) { ROS_INFO("I heard: [%s]", msg->data.c_str()); ros::Rate loop_rate(0.5);//block chatterCallback2() loop_rate.sleep(); } void multiThreadListener::chatterCallback2(const std_msgs::String::ConstPtr& msg) { ROS_INFO("I heard: [%s]", msg->data.c_str()); } int main(int argc, char **argv) { ros::init(argc, argv, "multi_sub"); multiThreadListener listener_obj; ros::MultiThreadedSpinner s(1);//表示使用几个线程 ros::spin(s);//利用多线程自动调用回调函数,这里多线程好像都是自动的调用回调函数的。 return 0; } /* ros::MultiThreadedSpinner spinner(4); // Use 4 threads spinner.spin(); // spin() will not return until the node has been shutdown */还没看呢!有空的时候补充
#include <thread>在这里值得注意的是回调函数的参数传递。
void chatterCallback(const std_msgs::String::ConstPtr& msg,type1 arg1, type2 arg2,...,typen argN) { ROS_INFO("I heard: [%s]", msg->data.c_str()); } int main(int argc, char** argv) { ros::Subscriber sub = n.subscribe<pkg::pkg.msg>("chatter", 1000, boost::bind(&chatterCallback,_1,arg1,arg2,...,argN); ///需要注意的是,此处 _1 是占位符, 表示了const std_msgs::String::ConstPtr& msg。pkg::pkg.msg 就是第一个参数的类型 具体实例代码如下(示例): #include<ros/ros.h> #include<guoqing/jisuan.h> #include<guoqing/xinxi.h> void messageCallback(const guoqing::xinxi::ConstPtr& msg, ros::ServiceClient &client, guoqing::jisuan &srv1) { if(msg->ok%10!=0) ROS_INFO("%s %d",msg->wenhou.c_str(),msg->ok); else { if(client.call(srv1)){ ROS_INFO("%s",msg->wenhou.c_str()); ROS_INFO("the resule of %d and %d is %d",(int)srv1.request.A,(int)srv1.request.B,(int)srv1.response.result); } else { ROS_ERROR("Failed to call service"); } } } int main(int argc,char **argv){ ros::init(argc,argv,"xuehao"); ros::NodeHandle n; ros::ServiceClient client=n.serviceClient<guoqing::jisuan>("cheng"); guoqing::jisuan srv1; srv1.request.A=3; srv1.request.B=5; ros::Subscriber sub = n.subscribe<guoqing::xinxi>("xunwen",10, boost::bind(&messageCallback,_1,client,srv1)); //ROS_INFO("rosok1111"); //ros::spin(); //ROS_INFO("rosok"); ros::spin(); return 0; }该处使用的url网络请求的数据。
提示:
参考文章spin,spinonce多线程订阅链接: link. ros中多参数回调的参考文章link
c++中的多线程编程link
