机器人学基础——正运动学(理论推导及c++实现)

机器人正运动学

机器人正运动学一般是指从机器人的关节位置到基于参考坐标系下末端执行器的位置。

平移变换和旋转变换

平移变换

假设我们有两个坐标系A和B,坐标系A与B的方位相同,xyz轴的指向都是一致的,即没有旋转变换。有一点p,则在A坐标系下从A坐标原点到p的位置矢量即为在B坐标系下p的位置矢量加上在A坐标系下B原点的位置矢量,公式如下
在这里插入图片描述

旋转变换

类似上述平移变换,我们假设坐标系A与坐标系B没有平移变换及原点位置一致,只有方位上的变换。如下
在这里插入图片描述
这个坐标系A和坐标系B原点一致并且在方位上不同,可以看成将坐标系A经过某个旋转变换进程为了B坐标系的方位。我们将这个变换的过程记为在这里插入图片描述
上标为参考坐标系,这个变换也可以解释为B坐标系在A坐标系下的方位

复合变换

当两个坐标系之间的关系坐标原点没有重合方位也不是一致的时候我们该怎么描述这个变换过程呢?
当坐标系A与B无法使用单独的平移变换或者单独的旋转变换来描述时,我们可以将这个过程看作是先经过平移后经过旋转或者先经过旋转再经过平移来描述。由于这个过程都是的参考系都是A坐标系即参考坐标系是固定的,所以先平移还是先旋转是没有区别的。

假设有一点p,在A坐标系下位置是在这里插入图片描述,在B坐标系下的描述为在这里插入图片描述,从坐标系A到B的旋转变换为在这里插入图片描述,坐标系A到B的平移变换为在这里插入图片描述

A到B的平移变换即为平移变换部分讲的在A坐标系下B的坐标原点的矢量。那么具有下面这个关系
在这里插入图片描述

数学描述

对平移变换和旋转的概念大致有了一个了解,那么具体应该应该怎么计算呢?不管是平移变换还是旋转变换我们都可以使用矩阵来描述这个过程。

平移矩阵

我们知道在三维空间中描述点或者矢量可以用x,y,z三个量来表达和描述。
而平移变换的本质上即为从坐标系A(参考坐标系)原点到坐标系B(目标坐标系)原点的一个矢量,所以平移矩阵的数学表达可以这样表示
在这里插入图片描述
[x,y,z],即为在A坐标系下B坐标系原点的位置

旋转矩阵

坐标系经过旋转变换之后,相对应的点的位置描述也要被改变,那么要怎么使用数学语言来描述这个变换呢
我们可以将从坐标系A到坐标系B的旋转变换分解为绕A(参考坐标系)系的X轴旋转α°,绕Y轴旋转β°,转z轴旋转γ°,三种基本的旋转,其他都可以看作是这三种基础旋转的叠加旋转。
使用矩阵来表示这三种旋转方式
绕x转α°:
在这里插入图片描述
绕y轴旋转:
在这里插入图片描述
绕z轴旋转
在这里插入图片描述

c++实现

根据以上基础理论我们大致了解了机器人正运动学中基础的旋转变换和平移变换,接下来我们就可以使用这些来进行机器人正解我们使用c++来完成这个过程,假设我们有一个urdf模型,那么第一步我们需要对这个模型进行解析

导入解析urdf模型

打开一个终端,进入之前下载的frank机械臂的工作空间内,我们新创建一个功能包frank_fk,用来表示正运动学求解的功能包

~/franka/src/franka_ros$ catkin_create_pkg franka_fk  rospy roscpp std_msgs

我们新建一个cpp

~/franka/src/franka_ros/franka_fk/src$ touch frank_fk.cpp
~/franka/src/franka_ros/franka_fk/include/franka_fk$ touch frank_fk.h

在panda机器人里面使用的是.urdf.xacore文件,我们需要对这个文件进行预处理变成urdf格式的文件才可以使用c++的接口对其进行读取解析
使用终端命令行将这个.urdf.xacore转换成urdf格式的文件

cd franka/
source ./devel/setup.bash 
rosrun xacro xacro src/franka_ros/franka_description/robots/panda/panda.urdf.xacro > panda.urdf

可以看到在franka下生成了一个panda.urdf文件,我们将它移动到panda.urdf.xacore路径下

使用vscode打开刚才创建的frank_fk
键入以下代码

franka_fk.h
#include <eigen3/Eigen/Dense>
#include <urdf/model.h>
#include <string>
#include <fstream>
#include <sstream>
#include <stdexcept>
#include <cstdlib>
#include <map>

class ROBOTFK
{
public:
    int Readmodel(std::string modelpath);
    Eigen::Matrix4d GetEndPos(std::map<std::string,double> jointname);
private:
    Eigen::Matrix3d GetJointpos();
    urdf::Model pdmodel;

};
franka_fk.cpp
#include <iostream>
#include <urdf_model/model.h>
#include <frank_fk.h>
#include <ros/ros.h>
int ROBOTFK::Readmodel(const std::string modelpath)
{
    int a = pdmodel.initFile(modelpath);
    std::cout << "Robot Name: " << pdmodel.getName() << std::endl;
    return a;
}
bool fileExists(const std::string& path) {
    std::ifstream file(path);
    return file.good();
}
int main()
{

    ROBOTFK robotfk;
    int success = robotfk.Readmodel("src/franka_ros/franka_description/robots/panda/panda.urdf");
    return 0;
}
cmakelist

cmake_minimum_required(VERSION 3.0.2)
project(franka_fk)

#add_compile_options(-std=c++11)
find_package(catkin REQUIRED COMPONENTS
  roscpp
  rospy
  urdf
  std_msgs
)

catkin_package(
 INCLUDE_DIRS include/franka_fk
)

include_directories(
  include/franka_fk
  ${catkin_INCLUDE_DIRS}
)

${catkin_EXPORTED_TARGETS})



add_executable(main src/frank_fk.cpp)

target_link_libraries(main
  ${catkin_LIBRARIES}
)

可以使用这个cpp暂时来进行urdf模型文件的读取,查看是否可以读取到正确的文件,接下来来进行一个简单的编译和运行

catkin_make
source ./devel/setup.bash
rosrun franka_fk main

可以看到运行结果是这样的

Robot Name: panda

这说明已经检测到正确的urdf文件并且读取出来我们的机器人名称为:panda

单关节旋转矩阵的计算

接下来我们对机器人单关节的变换矩阵进行c++的实现,需要从urdf模型中获取到我门制定关节对应的旋转轴(旋转类型)和相对上一关节位置的位移量,所以我们对Eigen::Matrix3d GetJointpos();这个函数进行填充


后面会补充…

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值