第六课 ROS的空间描述和变换

1、空间描述与变换

有两个坐标系A和B,B坐标系中有一个点P,如何把B坐标系中的P映射到A坐标系呢,这就涉及到空间描述与变换,

先看一下旋转矩阵:

上面中间的行向量中的元素表示在B坐标系当中的元素用A坐标系来表达。

 

是B坐标原点在A坐标系中的表达,它也是平移量;AP表示P在A坐标系中的表达,BP表示P在坐标系B中的表达。

要想将BP转换为A坐标系中的表达,就需要乘以旋转矩阵,它是旋转量。

下面目的是将B坐标系中的P点在A坐标系中进行表达

将它可以写成齐次坐标的形式如下,

中间的表达式可以成为旋转算子,

 

首先,看一下xyz固定角坐标系:

首先将一个坐标系B与一个已知坐标系A重合,先将B坐标系绕XA坐标系逆时针旋转γ角度,这里X轴坐标不变;然后再绕YA坐标轴逆时针旋转β角度;最后绕ZA轴逆时针旋转α角度。每一次旋转都是按照对应的固定参考系。我们将这种姿态的表示方法叫做X-Y-Z固定角坐标系。有时也叫偏航角,横滚角,俯仰角。

再来看一下欧拉表示法:

首先将一个坐标系B与一个已知坐标系A重合,首先将B坐标系绕ZA轴逆时针旋转α角度;再绕YA坐标轴逆时针旋转β角度;最后绕XA坐标系逆时针旋转γ角度。这个与上面的区别在于,每次旋转都是绕着坐标系B中的坐标轴旋转而不是绕着坐标系A中的坐标旋转。

最后来看一下四元素表示法:

与矩阵相比四元组更加高效,占用的空间更小,且便于插值。四元组表示复数,w+xi+yj+zk,其中i,j,k都是虚数单位。可以把四元组看做是一个标量和一个3维向量的组合;实部w表示标量,虚部(x,y,z)用v表示。其中w与旋转角度相关,v与旋转轴相关。

操作四元组,欧拉角或者X-Y-Z固定角可以用Bullet方法,KDL::Rotation方法等

tf

tf的原理

有三个坐标系,一个未知坐标系query,已知坐标系known,一个世界坐标系world

要求得未知坐标系在世界坐标系当中的表达,可以用旋转算子来表示:

未知坐标系在世界坐标系中的旋转算子=未知坐标系在已知坐标系中的旋转算子*已知坐标系在世界坐标系中的旋转算子,通过例子来看一下,

1、roscore

2、roslaunch turtle_tf turtle_tf_demo.launch

出现两只乌龟一只乌龟随着另外一只乌龟跑,

一只乌龟跟随另一只乌龟的运动而运动,这里面就包含三个坐标系,一个是乌龟1坐标系,一个是乌龟2坐标系,还有一个世界坐标系,

使用命令行工具,

view frames是以pdf的形式来展示运行框架的,

rosrun tf view_frames

打开该pdf,如下

turtle1和turtle2的父坐标系均为world坐标系。

然后再看,

下面看一下turtle1与turtle2作为之间的关系,前者作为父坐标系,后者作为子坐标系

再看一下tf_monitor工具

仍然以turtle1作为父框架,turtle2作为子框架

可以看到上面从turtle1到turtle2的变换的相关信息

TransformBroadcaster有一个方法sendTransform它可以提供的信息是已知坐标系到世界坐标系的旋转算子。

TransformListener提供了一个方法lookupTransform提供了一个未知坐标系到一个已知坐标系的旋转算子。

ExtrapolationException异常是两个frame之间存在请求,但是其中有一个frame是过时的,例如turtle1在listen时请求的是turtle2在50秒之前的请求,则会无效。

编写程序实现turtle tf_demo那个例子

打开eclipse,导入工程,然后新建一个源文件turtle_tf_broadcaster.cpp

#include<ros/ros.h>

#include<tf/transform_broadcaster.h>

#include<turtlesim/Pose.h>//turtlesim广播的是一个位姿消息,包括位置和角度

//定义一个全局变量,用于在命令行中输入turtle的名字

std::string turtle_name;

//回调函数的消息为turtlesim的msg

void posecallback(turtlesim::PoseConstPtr &msg)

{

  //创建一个TransformBroadcaster存储变换的相关信息

  static tf::TransformBroadcaster br;

  //接着创建一个对象Transform存储变换的相关信息,它包含旋转和位移的信息,有一个向量叫做Vector3表示坐标,以及四元组表示方向

  tf::Transform transform;

  //下面把turtlepose的相关信息copy到transform中,通过transform在tf树中进行广播,

  transform.setOrigin(tf::Vector3(msg->x,msg->y,0));

  //创建一个四元组

  tf::Quaternion q;

  //调用它的方法,因为它是绕着z轴旋转故有下面参数形式

   q.setRPY(0,0,msg->theta);

     //为transform的rotation赋值

  transform.setRotation(q);

  //然后再广播出去,创建一个transform::stamped对象

  br.sendTransform(tf::StampedTransform(transform,ros::Time::now(),"world",turtle_name));

}

int main(int agc,char **argv)

{

  ros::init(argc,argv,"turtle_tf_broadcaster");

  if(argc!=2)

  {

    ROS_ERROR("need turtle name as argument.");

    return -1;

  }

  turtle_name=argv[1];//传递进来的名字

  ros::NodeHandle node;

  //下面创建一个subscriber订阅对象,订阅的主题是turtle_name+"/pose",列表长度为10,回调函数为posecallback

  ros::Subscriber sub=node.subscribe(turtle_name+"/pose",10,&posecallback);

  ros::spin();

  return 0;

}

下面修改CMakeLisets.txt文件

add_executable(turtle_tf_broadcaster  src/turtle_tf_broadcaster.cpp)

target_link_libraries(turtle_tf_broadcaster ${catkin_LIBRARIES})

 

posted on 2017-02-02 12:12  gary_123  阅读(2143)  评论(0编辑  收藏  举报

导航