路径跟踪算法---Stanley Method实现

文章目录

  • 前言
  • 一、Stanley原理介绍
  • 二、主要代码实现
  • 三、效果


前言

Stanley Controller也是基于几何追踪的轨迹跟踪控制器,和Pure Pursuit不同的是,其基于前轮中心点为参考点进行控制,没有预瞄距离,以前轮中心点与最近参考轨迹点进行横向误差与导航角误差的计算。


一、Stanley原理介绍

在这里插入图片描述

如图,黑色曲线表示预设路径,红色方框表示车辆当前运动状态,偏离预设路径,使用Stanley Controller的目的是让车辆追踪预设路径,通过控制车的转向,使其靠近预设路径,图中黑色方框代表期望状态。图中红色箭头表示车辆实际运动方向,黑色箭头表示期望车辆运动方向,绿色虚线表示距离车辆最近路径点的切线,e_y 表示车与路径的最短距离,即横向偏差,θ e 为切线与实际运动方向的夹角,即航向偏差,δ e 表示期望运动方向与切线的夹角。

假定车辆实际位置在预设路径上,此时没有横向偏差,只需要转过 θe 角度即可,期望车辆转过的角度为:
δ = θ e θ e = θ r − θ x \ δ = θe \\ θe = θr - θx  δ=θeθe=θrθx
假定车辆实际运动方向与切线方向一致,此时没有航向偏差,只有横向偏差。车辆当前速度为v,t时间内运动距离为d,和速度存在一个系数k。
δ = δ e d = v / k t a n δ e = e y / d = k ∗ e y / v δ e = t a n − 1 ( k e y / v ) \ δ = δe \\ d = v / k \\tan δ e = e_y / d = k *e_y / v \\ δ e = tan^{-1}(k e_y/v)  δ=δed=v/ktanδe=ey/d=key/vδe=tan1(key/v)
同时,转向角不能任意,需要进行限幅处理,此时期望车转过的角度为(加入时间变化):
δ ( t ) = δ e ( t ) + θ e ( t ) = θ e ( t ) + t a n − 1 ( k e y ( t ) / v ( t ) ) , δ ( t ) ∈ [ δ m i n , δ m a x ] \ δ (t) = δ e(t) + θe(t) = θe(t) + tan^{-1}(k e_y(t)/v(t) ),δ (t) ∈ [δmin,δmax]  δ(t)=δe(t)+θe(t)=θe(t)+tan1(key(t)/v(t)),δ(t)[δmin,δmax]

二、主要代码实现

首先,需要发布路径,话题名为"/rc_path_pub",发布路径实现可参考博客读取机器人移动轨迹并在RVIZ界面中显示。
其次,作为测试仅仅将获取到的导航角偏差 δ ×一个比例系数作为角速度,预设的速度 v 作为线速度发送到底盘进行控制,因为用的模型是差速模型,没有进行角度限幅处理,实际想达到好的实验结果,可能需要将导航角偏差送到 pid 控制器进行实际的速度计算。代码参考知乎:Raiden

using path_type = std::pair<std::pair<double, double>, double>;//用amcl定位获取机器人位姿效果不好,故采用odom位姿,也可用其他定位方式void StanleyController::robotPoseCallBack(const geometry_msgs::PoseWithCovarianceStampedConstPtr& msg){// yaw =tf::getYaw(msg->pose.pose.orientation);// tf::pointMsgToTF(msg->pose.pose.position,current_pose);}void StanleyController::OdomCallback(const nav_msgs::Odometry::ConstPtr& odom_msg) {yaw =tf::getYaw(odom_msg->pose.pose.orientation);tf::pointMsgToTF(odom_msg->pose.pose.position,current_pose);}//两点间距离auto StanleyController::distance(path_type p1,path_type p2) -> double{return sqrt(pow(p1.first.first-p2.first.first,2) +pow(p1.first.second-p2.first.second,2));}//最近距离路径点索引auto StanleyController::GetMinDisIndex(path_type current_pose, std::vector<path_type> path) -> int{double min_dis = INFINITY;int idx = 0;for(int i = 0;i<path.size();i++){double dis  = distance(path[i],current_pose);if(dis < min_dis){min_dis = dis;idx = i;}}return idx;}void StanleyController::robotPathCallBack(const nav_msgs::PathConstPtr& msg){if(recv_path_flag){nav_path_.poses = msg->poses;recv_path_flag = false;start_pose_.position.x = nav_path_.poses[0].pose.position.x;start_pose_.position.y = nav_path_.poses[0].pose.position.y;path_vector.clear();for(int i = 0;i<nav_path_.poses.size();i++){path_type temp_path = std::make_pair(std::make_pair(nav_path_.poses[i].pose.position.x,nav_path_.poses[i].pose.position.y),tf::getYaw(nav_path_.poses[i].pose.orientation));path_vector.push_back(temp_path);}}}void StanleyController::run(){cmd_vel_pub = n_.advertise<geometry_msgs::Twist>("/cmd_vel",1);pose_sub = n_.subscribe<geometry_msgs::PoseWithCovarianceStamped>("amcl_pose",10,std::bind(&StanleyController::robotPoseCallBack,this,std::placeholders::_1));path_sub = n_.subscribe<nav_msgs::Path>("/rc_path_pub",10,std::bind(&StanleyController::robotPathCallBack,this,std::placeholders::_1));real_path_pub = n_.advertise<nav_msgs::Path>("rvizpath", 100, true);odom_pose_sub = n_.subscribe<nav_msgs::Odometry>("odom", 10, std::bind(&StanleyController::OdomCallback, this, std::placeholders::_1));ros::Rate loop_rate(10);recv_path_flag = true;path.header.frame_id = "/map";path.header.stamp = ros::Time::now();path_type init_pose = std::make_pair(std::make_pair(0.0,0.0),0.0);path_type now_pose = init_pose;while(ros::ok()){now_pose.first.first = current_pose.x();now_pose.first.second = current_pose.y();now_pose.second = yaw;//判断获取到路径if(path_vector.size()>0){int current_idx = GetMinDisIndex(now_pose,path_vector);path_type cur_path_point = path_vector[current_idx];//横向偏差double e_y = distance(now_pose,cur_path_point);e_y = (now_pose.first.second - cur_path_point.first.second) * cos(cur_path_point.second) - (now_pose.first.first - cur_path_point.first.first) * sin(cur_path_point.second) <=0 ? e_y : -e_y;//航向偏差,转到-PI 到 +PI之间double theta_e =  cur_path_point.second -now_pose.second ;  //参考点航向 - 自车航向角if(theta_e > M_PI){ theta_e = theta_e - 2* M_PI;} else if(theta_e < -M_PI){theta_e = theta_e + 2* M_PI;}//转角,限幅double delta =( theta_e + atan2(factor * e_y, speed) );// if(delta > M_PI/3){//     delta = M_PI/3;// } else if(delta < -M_PI/3){//     delta = -M_PI/3;// }//直接把转角当作角速度vel.linear.x = speed;vel.angular.z = delta * 2.5;cmd_vel_pub.publish(vel);}ros::spinOnce();loop_rate.sleep();}}

三、效果

最后,放一张效果图~~~
在这里插入图片描述


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/39762.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

天诚长租公寓智能门锁管理解决方案

人才是区域创新发展的第一资源&#xff0c;如何解决人才的住房问题&#xff0c;让人才“流进来”、“留下来”、“融进来”&#xff0c;就需要优先安排优质人才公寓、人才优租房和公共租赁住房房源&#xff0c;并为青年人才群体提供智能化、信息化的租住体验及通行服务。 一、…

四、(1)网络爬虫入门及准备工作(爬虫及数据可视化)

四、&#xff08;1&#xff09;网络爬虫入门及准备工作&#xff08;爬虫及数据可视化&#xff09; 1&#xff0c;网络爬虫入门1.1 百度指数1.2 天眼查1.3 爬虫原理1.4 搜索引擎原理 2&#xff0c;准备工作2.1 分析爬取页面2.2 爬虫拿到的不仅是网页还是网页的源代码2.3 爬虫就是…

app单页下载页源码带管理后台

新版带后台管理APP应用下载页,自动识别安卓苹果下载页&#xff0c;带管理后台&#xff0c;内置带3套App下载模板带中文模板/英文模板随时切换。 app单页下载页源码带管理后台

告别PS修图,设计师都在用的AI抠图工具

引言 大家好&#xff01;如果你是美工或设计师&#xff0c;肯定深知Photoshop修图的繁琐和耗时。现在有一款超方便的工具&#xff0c;让你摆脱这些问题——千鹿设计助手。它不仅是个抠图工具&#xff0c;还能通过先进的AI技术&#xff0c;让抠图变得简单快速&#xff0c;让你专…

[Leetcode 128][Medium] 最长连续序列

目录 题目描述 整体思路 具体代码 题目描述 原题链接 整体思路 首先看到找连续升序排序的最长序列长度&#xff0c;想到对数组进行排序预处理。但是排序算法时间复杂度需要O(nlogn)&#xff0c;题目要求时间复杂度为O(n)。因此不能进行排序与处理 接着想到数据结构哈希表&a…

为什么网上商店需要翻译成其他语言

网上商店不仅仅是一个可以买到商品的网站。它是一个完整的电子商务平台&#xff0c;为来自世界各地的用户提供购买所需物品的机会。但是&#xff0c;为了让这些用户舒适地使用网站&#xff0c;需要高质量的翻译和本地化。 本地化是指产品或服务适应特定文化或市场的过程。它包…

Nginx详解-安装配置等

目录 一、引言 1.1 代理问题 1.2 负载均衡问题 1.3 资源优化 1.4 Nginx处理 二、Nginx概述 三、Nginx的安装 3.1 安装Nginx 3.2 Nginx的配置文件 四、Nginx的反向代理【重点】 4.1 正向代理和反向代理介绍 4.2 基于Nginx实现反向代理 4.3 关于Nginx的location路径…

CSS 文本输入框右下角的尺寸控件(三斜线:-webkit-resizer)消除,以及如何配置其样式,添加 resize 让标签元素可进行拖拽放大。

前言&#xff1a;在日常的前端开发中&#xff0c;不管是原始的和 还在在各类组件库中的文本输入框中&#xff0c;元素内容的右下角总是有一个三斜线的样式&#xff0c;本文简单了解它是什么&#xff1f;如何去控制并修改样式&#xff1f; 一、它是&#xff1f; 这三个斜线其实…

echarts实现3D柱状图(视觉层面)

一、第一种效果 效果图 使用步骤 完整实例&#xff0c;copy就可直接使用 <template><div :class"className" :style"{height:height,width:width}" /> </template><script>import echarts from echartsrequire(echarts/theme/…

Qt通过句柄获取其它进程控件实例

1.通过spy获取想要获取控件的句柄id 通过spy获取另一个软件的文本框的句柄 2.Qt写代码&#xff0c; 根据句柄获取文本框的内容 void getTextFromExternalWindow(HWND hwnd) {const int bufferSize 256;TCHAR buffer[bufferSize];// 获取窗口文本内容int length GetWindowT…

一文读懂企业为什么需要数字工厂管理系统

在当今这个日新月异的数字化时代&#xff0c;企业面临着前所未有的挑战与机遇。为了在激烈的市场竞争中保持领先地位&#xff0c;实现高效、灵活、可持续的生产运营&#xff0c;企业纷纷转向数字化转型&#xff0c;而数字工厂管理系统作为其中的关键一环&#xff0c;正逐步成为…

京东e卡怎么用?

京东618过去后&#xff0c;就没有多大购物欲望了&#xff0c;最后导致我手里还有好几张200块钱面值的e卡没地方用 本来说送朋友&#xff0c;但是又感觉面值太小了 最后还是在收卡云上把提取出来了&#xff0c;主要回收价格不错&#xff0c;而且到账也快&#xff0c;很方便

VMware配置Ubuntu

VMware下载官方链接&#xff1a;Download VMware Workstation Player | VMware Ubuntu20.04下载&#xff1a;https://ubuntu.com/download/desktop 安装步骤 点击【浏览】可更改安装位置&#xff08;建议不要安装在C盘&#xff0c;可以在D盘或其它磁盘下新建一个“ubuntu”文…

linux深度deepin基于rsync和apt-mirror同步软件源及构建本地内网源

目录 一、rsync方式二、apt-mirror方式1.安装apt-mirror2.配置apt-mirror(/etc/apt/mirror.list)3.新建存放目录开始下载 3.发布mirror站点 一、rsync方式 参考官方文档地址&#xff1a; https://www.deepin.org/index/docs/wiki/05_HOW-TO/08_%E9%95%9C%E5%83%8F%E5%8A%A0%E9%…

时钟服务器方案选型推荐:ATGM332D-5T和ATGM331C-5T

ATGM331C-5T系列模块同样是具有高灵敏度、低功耗、低成本等优势&#xff0c;适用于电力授时设备、时钟服务器、守时设备&#xff0c;可以直接替换Ublox LEA T系列模块。 性能指标&#xff1a; 从下面的图来看&#xff0c;ATGM331C-5T系列比ATGM332D-5T系列性能更好&#xff0c;…

民宿小程序开发,在线预订模式

一、开发背景 如今&#xff0c;随着互联网技术的快速发展&#xff0c;大众的生活消费都集中在了手机上&#xff0c;通过手机进行各种活动&#xff0c;同时也包括了预订酒店民宿&#xff0c;由此&#xff0c;民宿预约小程序出现在了大众的生活中。 二、民宿小程序特点 民宿小…

中国经济昆虫志(55卷)

中国经济昆虫志&#xff0c;共55卷&#xff0c;内容包括概述、形态特征、分类等。各级分类单元均编有检索表&#xff0c;每个种有特征描述、地理分布&#xff0c;有的还记载有生活习性和防治方法。为便于鉴定&#xff0c;绘制有特征图和彩色图。 包括鞘翅目天牛科、半翅目蝽科、…

Python创建异步任务队列库之Huey使用详解

概要 Huey 是一个简单的 Python 库,用于创建异步任务队列。它的设计目标是简单易用,同时具备强大的功能。Huey 可以轻松地将任务添加到队列中,然后在后台线程中处理这些任务,从而避免阻塞主线程。这使得 Huey 非常适合处理 I/O 密集型或长时间运行的任务。此外,Huey 还支…

Qt安装配置教程

目录 一、下载Qt二、进行安装1、点击安装包&#xff08;QT6.7版本演示&#xff09;2、注册Qt账号3、选择安装的位置4、选择对应的组件 三、新建项目1、打开Qt Creator2、创建项目3、编辑名称和地址4、选择默认的CMake或切换成qmake构建5、选择自己的编译器&#xff0c;在此选择…

从.mat文件中导入数据到simulink进行FFT分析

1. 在matlab中准备数据 .mat 文件中包含时间向量和需要分析的数据 load(fcssiabc061302.mat);提取时间和需要分析的数据 time fcssiabc061302.X.Data; % 时间向量 signal fcssiabc061302.Y(1).Data; % A相电流数据 将数据转换为“structure with time”格式…