2020-10-05

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档


ROS节点、服务、话题。

对于发布者来说,需要实现在发布话题的同时,提供一个服务,这时会出现ros::spin()与ros::spinonce()的冲突。经验证,服务与订阅者的回调函数只有在持续回调的过程中才有用。
对于接收者需要实现订阅的话题消息满足一定条件后,请求一个服务,这同样出现ros::spin()与client.call(srv)的冲突。


提示:以下是本篇文章正文内容,下面案例可供参考

一、发布节点的实现

对于服务节点的冲突在查阅相关资料后一般有两种思路。(首先说明一下对于多回调函数的处理方式)
一个节点中含有多回调函数时,通常使用的方法是定义一个类,回调函数在这个类中读取。然后在ros::spin()一下即可。
这种方法请看链接: link.
另一种方法就是通过多线程的方式实现的。
在ros中多线程可以分为同步线程和异步线程

1.同步线程

个人感觉其想要解决的问题是多回调函数的问题,其实现方法如下:

#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
*/

2.异步线程

/*ros::AsyncSpinner spinner(4); // Use 4 threads
spinner.start();
ros::waitForShutdown();*/
#include<ros/ros.h>
#include<guoqing/xinxi.h>
#include<guoqing/jisuan.h>
#include<boost/thread.hpp>


bool add(guoqing::jisuan::Request &req,
        guoqing::jisuan::Response &res)
{
    res.result=req.A*req.B;
    ROS_INFO("the resule of %d and %d is %d",(int)req.A,(int)req.B,(int)res.result);
    return true;   
}

int main(int argc,char **argv){
    ros::init(argc,argv,"xingming");
    ros::NodeHandle n;
    ros::Publisher pub=n.advertise<guoqing::xinxi>("xunwen",10);
    ros::ServiceServer service=n.advertiseService("cheng",add);
    ros::AsyncSpinner s(1);
    s.start();
    ros::Rate loop_rate(1);
    long int i=0;
    while(ros::ok()){
        i++;
        guoqing::xinxi msg1;
        msg1.wenhou="what are you doing";
        msg1.ok=i;
        if(i%10==0){
            msg1.wenhou="ok";
            //msg1.ok=1;
            pub.publish(msg1);
            ROS_INFO("%s",msg1.wenhou.c_str());
            ros::spinOnce();
            loop_rate.sleep();
        }
        else{
            pub.publish(msg1);
            ros::spinOnce();
            loop_rate.sleep();
        }
    }
    return 0;
}

3.c++中自带的多线程使用方法

还没看呢!有空的时候补充

#include <thread>

二、订阅节点的实现

1.回调函数的多参数应用

我是在subscribe的回调函数中使用了多参数传递的方式。

在这里值得注意的是回调函数的参数传递。


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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值