roscpp模式总结

综述

在写代码的过程中,一直苦于没有一个比较通用标准的代码模版。为了形成一个良好的代码规范,所以在这里博主把这段时间写的代码提取出来,写成下面的模版。希望以后可以提高编写roscpp的效率。


1. Publisher

代码模版展示:

#include "ros/ros.h"

#include "std_msgs/String.h"//某些message

#include <tf/transform_listener.h>//某些ros中使用的包

#include <sstream>//某些cpp中使用的库

int main(int argc, char **argv){
ros::init(argc,char);
ros::NodeHandle n;
ros::Publisher chatter_pub=n.advertise<std_msgs::String>("chatter",1000);
ros::Rate r(10);
int count=0;
while(ros::ok()){
std_msgs::String msg;//初始化一个msg

std::stringstream ss;
ss<<"hello world"<<count;
chatter_pub.publish(msg);

ros::spinOnce();
r.sleep();
++count;

}
return 0;
}

这里的publisher是用来发送topic“chatter”的。

2.subscriber

#include "ros/ros.h"
#include "std_msgs/Int32.h"
#include "geometry_msgs/Pose.h"
#include "basic_msgs/gridpose.h"//自定义message
#include <time.h>
#include "math.h"

void callback(const geometry_msgs::Pose::ConstPtr &msg){
//可以用全局变量保存想要保存的内容

}

int main(int argh, char **argv){
ros::init(argh, argue, "talker");
ros::NodeHangle n;
ros::Subscriber sub=n.subscribe("robot_pose",2,callback);
ros::Rate r(2);
int count =0;
int i=1;
while( ros::ok()){
std_msgs::Int32 msg;//初始化msg
//可能会有pub.publish(<message>);
ros::spinOnce();
r.sleep();
}
return 0;

}

3.service

分为两种,一种是serviceserver,即为提供service的,一种是serviceclient,请求得到响应的客户端。

3.1 serviceserver

#include "ros/ros.h"
#include "std_msgs/Int32.h"
#include "operation/mapping_status.h"//同一目录下的自定义service
#include <iostream>


bool server(operation::mapping_status::Request &req, operation::mapping_status::Response &res){
    ...

}

int main(int argc, char **argv){
    ros::init(argc,argv,"mapping_status_node");
    ros::NodeHandle n;

    ros::param::param<std::string>("mapsaver_bringup_path",mapsaver_bringup_path,"/.../path/file");//ros param
    ros::ServiceServer service=n.advertiseService("/mapping_status",server);//ros service server.

    ros::spin();

    return 0;


}

3.2 serviceclient

#include "ros/ros.h"
#include "operation/mapping_status.h"
#include <cstdlib>

int main(int argc, char **argv)
{
    ros::init(argc,argv,"mapping_status_client");
    ros::NodeHandle n;
    ros::ServiceClient client=n.serviceClient<opeartion::mapping_status>("/mapping_status");
    srv.request.data=atoll(argv[1]);//第一个值为request的输入值
    if(client.call(srv)){

    ROS_INFO("successed");
    }else{
    ROS_INFO("Failed");

    }

    return 0;

}

4.transform

分为两种:broadcaster和listener.

4.1 broadcaster:

#include <ros/ros.h>
#include <tf/transform_broadcaster.h>

int main(int argc, char** argv){
    ros::init(argc, argv, "my_tf_broadcaster");
    ros::NodeHandle n;
    tf::TransformBroadcaster br;

    ros::Rate r(20);
    while(n::ok()){
      tf::Transform transform;
      transform.setOrigin(tf::Vector3(x,y,z));//x,y,z 自己添
      transform.setRotation(tf::Quaternion(q_x,q_y,q_z,q_w));//同理
      br.sendTransform(tf::StampedTransform(transform,ros::Time::now(),"parent frame","child frame"));//child frame 在parent frame 的位置就是上面的姿态值。
      ros::spinOnce();


    }
    return 0;
};

4.2 listener:

#include <ros/ros.h>
#include <tf/transform_listener.h>
#include <geometry_msgs/Pose.h>

geometry_msgs::Pose pose_listen(tf::StampedTransform &transform_listener){
    geometry_msgs::Pose pose;
    pose.position.x=transform_listener.getOrigin().x()*20;
    pose.position.y=transform_listener.getOrigin().y()*20;
    ...
    return pose;

}


int main(int argc,char** argv){
    ros::init(argc,argv,"my_tf_listener");
    ros::NodeHandle n;
    tf::TransformListener listener;
    ros::Rate r(1);
    while(n.ok()){
    tf::StampedTransform transform_listener;

    try{
    listener.waitForTransform("/frame1","/frame2",ros::Time(0),ros::Duration(10.0));//ros::Time(0)即使时间可能不同步,但是不会报错,如果是ros::Time.now()由于延时,会发生报错的。
    listener.loopupTransform("/frame1","/frame2",ros::Time(0),transform_listener);//frame1是主frame
    }catch(tf::TransformException ex){
    ROS_ERROR("%s",ex.what());
    ros::Duration(1.0).sleep();
    continue;
    }

    geometry_msgs::Pose pose;//对于之前得到的转换进行处理
    pose=pose_listener(transform_listener);

    ros::spinOnce();
    r.sleep();
    }
    return 0;
}

4.3 transform point

最后,是让点转换到某个坐标系中去。代码如下:


#include <ros/ros.h>
#include <geometry_msgs/PointStamped.h>
#include <tf/transform_listener.h>
#include <basic_msgs/laser_points.h>
#include <basic_msgs/points.h>
#include <vector>
#include <iostream>

class Scan_info{
public:
    sensor_msgs:LaserScan ranges;

    Scan_info();
    ~Scan_info();
    void scancallback(const sensor_msgs::LaserScanConstPtr &msg);


};

Scan_info::Scan_info(){
    ROS_INFO("Constructor started");
}

Scan_info::~Scan_info(){
    ROS_INFO("Deconstructor started");
}

void Scan_info::scancallback(const sensor_msgs::LaserScanConstPtr &msg){

    Scan_info::laser_point_vector.data.clear();
    ...

    Scan_info::laser_point_vector.data.push_back(test_msg);//这是一个vector,会存到实例化后的对象上。
}

void transformPoint(const tf::TransformListener& listener, geometry_msgs::PointStamped& laser_point,...,int i){
    ...
    try{
    geometry_msgs::PointStamped base_point;
    base_msgs::points point;
    listener.transformPoint("picture_frame",laser_point,base_point);//得到base_point.
    }catch(tf::TransformException& ex){
    ROS_ERROR("...");
    }
}

int main(int argc,char** argv){
    ros::init(argc,argv,"robot_tf_listener");
    ros::NodeHandle n;
    Scan_info *scaninfo=new Scan_info();
    ros::Subscriber scan_sub=n.subscribe("base_scan",100,&Scan_info::scancallback,scaninfo);//这里的scaninfo是实例化的类,这样subsriber里面就可以返回值到主函数里面了。

    tf::TransformListener listener(ros::Duration(10));
    ros::Timer timer1=n.createTimer(ros::Duration(1.0),boost::bind(&transformPoint,boost::ref(listener),boost::ref(scaninfo->laser_point),int i);//这里的boost::ref里面是对象的静态数据成员,或者对象。boost::bind的第一个参数必须是&transformPoint,来作为transformpoint函数处理点的转换。

}

5.关于CMakelists与package.xml

原则:不要在一个workspace里面自定义了message,srv,同时在不同的package里面的cpp里面使用他们。如果使用,可以先拿出cpp的package,catkin_make workspace里面的内容,然后再把包含这些cpp的package放进去,进行编译。

需要生成新的message包里面的package.xml中,需要加入

<build_depend>message_generation</build_depend>
<run_depend>message_runtime</run_depend>

CMakelists
里面

find package(catkin REQUIRED COMPONENTS
roscpp
...
message_generation
)

generate_messages(
DEPENDENCIES
std_msgs
geometry_msgs
)

catkin_package(
CATKIN_DEPENDS message_runtime
)

6.linux管理

在ros中,有时候需要在cpp中调用系统命令行。采用多线程方法,在子线程中调用命令行,而在别的线程中仍继续运行,等待service从客户端调用。
举个例子

int parameter =-1;
pthread_t id;
int i,ret;
struct Param{
int type;

};

void *thread(void *ptr){
    Param *param=(Param *)ptr;//强制类型转换
    //...
    //子线程程序

}

bool server(...){

    //这里在一个serviceserver里面用了多线程
    Param *param=new Param();
    param->type=req.data;
    ret=pthread_create(&id,NULL,thread,param);//进入子线程程序
    //。。。主线程
    sleep(1);
    return true

综上


  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值