飞道的博客

差动驱动机器人轨迹-CoCube

340人阅读  评论(0)

轨迹博客:

玫瑰线轨迹如何规划?(desmos+ROS2+turtlesim+……)

ROS1云课→23turtlesim绘制小结(数学和编程)


如上所涉及的机器人假定模型都是差动驱动机器人。

许多移动机器人使用一种称为差动驱动的驱动机构。它由安装在公共轴上的两个驱动轮组成,每个轮可以独立地向前或向后驱动。

图1:差动驱动运动学-Dudek和Jenkin《移动机器人的计算原理》

机器人旋转的点被称为ICC -瞬时曲率中心

通过改变两个轮子的速度,可以改变机器人的轨迹。

这类驱动器有三个有趣的例子:

  1. 直线运动,左右轮速度一样
  2. 绕轮轴的中点旋转,也就是原地旋转,左右轮速度大小一样,方向相反。
  3. 绕某一轮为中心旋转,左轮或右轮只有一轮有速度

注意,差动驱动机器人不能沿着轴的方向移动——这是一个奇点

曲线上存在不可导、不连续、根本没有定义的点,这些点就叫做奇点。基本上来说求导就可以瑕点是函数趋于无穷的点;奇点是函数未定的点。比如间断点,无定义点。奇点包含瑕点。

机器人系统数学建模(现代控制理论1)

图2 差动机器人数学建模

差速驱动车辆对每个车轮速度的微小变化非常敏感。轮子之间相对速度的差异会影响机器人的轨迹。它们对地平面的微小变化也非常敏感,可能需要额外的轮子(脚轮)来支撑。

凹凸不平的地面会影响轨迹精度。

差动驱动机器人的正向运动学

在图1中,假设机器人在某个位置(x,y),朝向与X轴成θ角的方向。假设机器人的中心位于轮轴的中点。通过操纵控制参数Vl、Vr,可以使机器人移动到不同的位置和方向。(注:Vl,Vr)为车轮沿地面的速度)。

图3:差动机器人的正向运动学 

移动机器人的逆运动学

如何控制机器人达到给定的位置(x,y,θ)——这就是所谓的逆运动学问题。

不幸的是,差动驱动机器人在建立其位置时符合所谓的非完整约束。例如,机器人不能沿着它的轴横向移动。类似的非完整约束是汽车只能转动前轮。它不能直接侧向移动,因为平行泊车(侧方位停车)需要更复杂的转向操作。因此,不能简单地指定一个任意的机器人姿态(x,y,θ)并找到控制机器人所需要的速度。

这激活了机器人沿直线移动,然后原地旋转一圈,然后再次直线移动的策略,作为差动驱动机器人的导航策略。

直线运动轨迹:

圆周运动轨迹: 

未闭合

 闭合后,但控制为开环,也就是没有反馈控制。

正方形运动轨迹(非连续控制): 

 可参考官方示例代码:


  
  1. #include <rclcpp/rclcpp.hpp>
  2. #include <turtlesim/msg/pose.hpp>
  3. #include <geometry_msgs/msg/twist.hpp>
  4. #include <std_srvs/srv/empty.hpp>
  5. #include <math.h>
  6. turtlesim::msg::Pose::SharedPtr g_pose;
  7. turtlesim::msg::Pose g_goal;
  8. enum State
  9. {
  10. FORWARD,
  11. STOP_FORWARD,
  12. TURN,
  13. STOP_TURN,
  14. };
  15. State g_state = FORWARD;
  16. State g_last_state = FORWARD;
  17. bool g_first_goal_set = false;
  18. #define PI 3.141592
  19. void poseCallback(const turtlesim::msg::Pose::SharedPtr pose)
  20. {
  21. g_pose = pose;
  22. }
  23. bool hasReachedGoal()
  24. {
  25. return fabsf(g_pose->x - g_goal.x) < 0.01 && fabsf(g_pose->y - g_goal.y) < 0.01 && fabsf(g_pose->theta - g_goal.theta) < 0.002;
  26. }
  27. bool hasStopped()
  28. {
  29. return g_pose->angular_velocity < 0.0001 && g_pose->linear_velocity < 0.0001;
  30. }
  31. void printGoal()
  32. {
  33. RCLCPP_INFO(rclcpp:: get_logger( "draw_square"), "New goal [%f %f, %f]", g_goal.x, g_goal.y, g_goal.theta);
  34. }
  35. void commandTurtle(rclcpp::Publisher<geometry_msgs::msg::Twist>::SharedPtr twist_pub, float linear, float angular)
  36. {
  37. geometry_msgs::msg::Twist twist;
  38. twist.linear.x = linear;
  39. twist.angular.z = angular;
  40. twist_pub-> publish(twist);
  41. }
  42. void stopForward(rclcpp::Publisher<geometry_msgs::msg::Twist>::SharedPtr twist_pub)
  43. {
  44. if ( hasStopped())
  45. {
  46. RCLCPP_INFO(rclcpp:: get_logger( "draw_square"), "Reached goal");
  47. g_state = TURN;
  48. g_goal.x = g_pose->x;
  49. g_goal.y = g_pose->y;
  50. g_goal.theta = fmod(g_pose->theta + static_cast< float>(PI) / 2.0f, 2.0f * static_cast< float>(PI));
  51. // wrap g_goal.theta to [-pi, pi)
  52. if (g_goal.theta >= static_cast< float>(PI)) g_goal.theta -= 2.0f * static_cast< float>(PI);
  53. printGoal();
  54. }
  55. else
  56. {
  57. commandTurtle(twist_pub, 0, 0);
  58. }
  59. }
  60. void stopTurn(rclcpp::Publisher<geometry_msgs::msg::Twist>::SharedPtr twist_pub)
  61. {
  62. if ( hasStopped())
  63. {
  64. RCLCPP_INFO(rclcpp:: get_logger( "draw_square"), "Reached goal");
  65. g_state = FORWARD;
  66. g_goal.x = cos(g_pose->theta) * 4 + g_pose->x;
  67. g_goal.y = sin(g_pose->theta) * 4 + g_pose->y;
  68. g_goal.theta = g_pose->theta;
  69. printGoal();
  70. }
  71. else
  72. {
  73. commandTurtle(twist_pub, 0, 0);
  74. }
  75. }
  76. void forward(rclcpp::Publisher<geometry_msgs::msg::Twist>::SharedPtr twist_pub)
  77. {
  78. if ( hasReachedGoal())
  79. {
  80. g_state = STOP_FORWARD;
  81. commandTurtle(twist_pub, 0, 0);
  82. }
  83. else
  84. {
  85. commandTurtle(twist_pub, 1.0f, 0);
  86. }
  87. }
  88. void turn(rclcpp::Publisher<geometry_msgs::msg::Twist>::SharedPtr twist_pub)
  89. {
  90. if ( hasReachedGoal())
  91. {
  92. g_state = STOP_TURN;
  93. commandTurtle(twist_pub, 0, 0);
  94. }
  95. else
  96. {
  97. commandTurtle(twist_pub, 0, 0.1f);
  98. }
  99. }
  100. void timerCallback(rclcpp::Publisher<geometry_msgs::msg::Twist>::SharedPtr twist_pub)
  101. {
  102. if (!g_pose)
  103. {
  104. return;
  105. }
  106. if (!g_first_goal_set)
  107. {
  108. g_first_goal_set = true;
  109. g_state = FORWARD;
  110. g_goal.x = cos(g_pose->theta) * 4 + g_pose->x;
  111. g_goal.y = sin(g_pose->theta) * 4 + g_pose->y;
  112. g_goal.theta = g_pose->theta;
  113. printGoal();
  114. }
  115. if (g_state == FORWARD)
  116. {
  117. forward(twist_pub);
  118. }
  119. else if (g_state == STOP_FORWARD)
  120. {
  121. stopForward(twist_pub);
  122. }
  123. else if (g_state == TURN)
  124. {
  125. turn(twist_pub);
  126. }
  127. else if (g_state == STOP_TURN)
  128. {
  129. stopTurn(twist_pub);
  130. }
  131. }
  132. int main(int argc, char** argv)
  133. {
  134. rclcpp:: init(argc, argv);
  135. auto nh = rclcpp::Node:: make_shared( "draw_square");
  136. auto pose_sub = nh-> create_subscription<turtlesim::msg::Pose>( "turtle1/pose", 1, std:: bind(poseCallback, std::placeholders::_1));
  137. auto twist_pub = nh-> create_publisher<geometry_msgs::msg::Twist>( "turtle1/cmd_vel", 1);
  138. auto reset = nh-> create_client<std_srvs::srv::Empty>( "reset");
  139. auto timer = nh-> create_wall_timer(std::chrono:: milliseconds( 16), [twist_pub](){ timerCallback(twist_pub);});
  140. auto empty = std:: make_shared<std_srvs::srv::Empty::Request>();
  141. reset-> async_send_request(empty);
  142. rclcpp:: spin(nh);
  143. }

开环控制,重复精度不高,效果如下:

 

思考题:

如何绘制如下曲线,选择一款绘制即可。

 


-Fin-


 

 

 

 


转载:https://blog.csdn.net/ZhangRelay/article/details/128210755
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场