yocs_velocity_smoother速度平滑库知识

news2024/10/1 19:22:00

一.C++

(1)nth_element()用法

头文件:#include<algorithm>

nth_element:在数组或容器中将第几大或小的元素放到该放的位置上。(默认第几小,可以用cmp自定义为第几大)

如:nth_element(a,a+6,a+10);  //在a中,找到 “第七小”将其放到a[6].

例如:

int array[5] = {2 4 6 8 10};

上面这个序列中第3小的是6,第4小的是8,经过nth_element()函数之后可以把第n小的元素K移动到第n的位置,并且位于K之前的位置比K小,位于K字后的位置比K大,nth_element()函数默认是升序排序,可以自定义降序排序。

#include<iostream>
#include<algorithm>
#include<vector>
using namespace std;
int cmp(int a,int b)
{
	return a>b;
}
int main()
{
	int a[5]={2,4,6,8,10};
	
	nth_element(a,a+2,a+5);  //在a中,找到 第三小 
	for(int i=0;i<5;i++)
	{
		cout<<a[i]<<" ";
	}
	cout<<endl;
	cout<<"a中第3小:"<<a[2]<<endl;
	
	 nth_element(a,a+2,a+5,cmp);
	for(int i=0;i<5;i++)
	{
		cout<<a[i]<<" ";
	}
	cout<<endl;
    	cout<<"a中第3大为:"<<a[2]<<endl;
	
	return 0;
}
//排序规则采用默认的升序排序
void nth_element (RandomAccessIterator first,
                  RandomAccessIterator nth,
                  RandomAccessIterator last);
//排序规则为自定义的 comp 排序规则
void nth_element (RandomAccessIterator first,
                  RandomAccessIterator nth,
                  RandomAccessIterator last,
                  Compare comp);

其中,各个参数的含义如下:

  • first 和 last:都是随机访问迭代器,[first, last) 用于指定该函数的作用范围(即要处理哪些数据);
  • nth:也是随机访问迭代器,其功能是令函数查找v_y_inc“第 nth 大”的元素,并将其移动到 nth 指向的位置;
  • comp:用于自定义排序规则。

注意,鉴于 nth_element() 函数中各个参数的类型,其只能对普通数组或者部分容器进行排序。换句话说,只有普通数组和符合以下全部条件的容器,才能使用使用 nth_element() 函数:

  1. 容器支持的迭代器类型必须为随机访问迭代器。这意味着,nth_element() 函数只适用于 array、vector、deque 这 3 个容器。
  2. 当选用默认的升序排序规则时,容器中存储的元素类型必须支持 <小于运算符;同样,如果选用标准库提供的其它排序规则,元素类型也必须支持该规则底层实现所用的比较运算符;
  3. nth_element() 函数在实现过程中,需要交换某些元素的存储位置。因此,如果容器中存储的是自定义的类对象,则该类的内部必须提供移动构造函数和移动赋值运算符。

 velocity_smoother_nodelet.hpp程序代码

 //返回vector的中位数
double median(std::vector<double> values) {
    // Return the median element of an doubles vector
    nth_element(values.begin(), values.begin() + values.size()/2, values.end());
    return values[values.size()/2];
  };

(2)C++智能指针成员函数reset

头文件:#include<memory>

.............
typedef boost::shared_ptr< ::geometry_msgs::Twist > TwistPtr;
typedef boost::shared_ptr< ::geometry_msgs::Twist const> TwistConstPtr;
..........

geometry_msgs::Twist   target_vel;
geometry_msgs::TwistPtr cmd_vel; 

//target_vel初始化
void VelocitySmoother::velocityCB(const geometry_msgs::Twist::ConstPtr& msg)
{
    ......
    target_vel.linear.x  =  msg->linear.x;
    target_vel.linear.y  =  msg->linear.y;
    target_vel.angular.z =  msg->angular.z;
    ......
}

void VelocitySmoother::spin()
{
    ......
    cmd_vel.reset(new geometry_msgs::Twist(target_vel));
    ......
}

cmd_vel是智能指针对象(shared_ptr< ::geometry_msgs::Twist >)

 cmd_vel.reset(new geometry_msgs::Twist(target_vel));

令cmd_vel智能指针存放指针target_vel,也就是指向target_vel的空间,并且释放原来的空间。

(3)boost::thread

boost::thread*  worker_thread_;
......
worker_thread_ = new boost::thread(boost::bind(&VelocitySmoother::spin, this));

 创建一个线程

boost::thread(threadFun);

参数:可以是函数对象或者函数指针,并且该函数无参数,返回void类型。

(待补充)

(4)boost::bind()使用方法

int f(int a,int b)
{
    return a+ b;
}

int g(int a,int b, int c)
{
    return a + b + c;
}

boost::bind(f,1,2)可以产生无参数函数对象,f(1,2);boost::bind(g,1,2,3)可以产生无参函数g(1,2,3)

其中boost::bind()中使用较为频繁的还有占位符:

如:boost::bind(f, _1, 5) (x) 中,_1就是一个占位符,其位于f函数形参的第一形参 int a 的位置,5位于第二形参 int b 的位置;

_1 表示(x)参数列表的第一个参数;所以,boost::bind(f, _1, 5) (x) 相当于 f(x ,5)。再看下面这些例子:

boost::bind(f, _2, _1)(x, y);   //相当于f(y,x),即_2表示后面参数列表的第二个位置:y
boost::bind(g, _1, 9, _1)(x);  //相当于g(x, 9, x)
boost::bind(g, _3, _3, _3)(x, y, z);  //相当于g(z, z, z)

ROS中常用的boost::bind()

ROS编程过程中,有许多需要给回调函数传递多个参数的情况

回调函数只有一个参数

#include <ros/ros.h>
#include <turtlesim/Pose.h>

//单个参数为:消息类型为turtlesim::Pose的常量指针msg
void callback_one(const turtlesim::PoseConstPtr& msg)  
{
    float pose_x;
    pose_x = msg->x;
    ROS_INFO("x = [%f]",pose_x);  //打印出小乌龟所在的x坐标
}

回调函数有多个参数

#include <ros/ros.h>
#include <turtlesim/Pose.h>

//三个参数:常量指针msg、x、y
void callback_more(const turtlesim::PoseConstPtr& msg, int x, int y)  
{
    float pose_x;
    pose_x = msg->x;
    ROS_INFO("x = [%f]",pose_x);  //打印出小乌龟所在的x坐标
    ROS_INFO("input_x = [%i] ; input_y = [%i]", x, y);  //打印出输入的参数x、y
}

 主函数

int input_x = 1;
int input_y = 2;

int main(int argc, char** argv)
{
    ros::init(argc, argv, "test");
    ros::NodeHandle n;
    
    ros::Subscriber pose_sub = n.subscribe<turtlesim::Pose>("/turtle1/pose_one", 10, callback_one);
    //回调函数为单个参数时,这里直接使用callback,传递的单个参数为:接收话题名为/turtle1/pose_one中的内容;
    
    ros::Subscriber pose_sub = n.subscribe<turtlesim::Pose>("/turtle1/pose_more", 10, boost::bind(&callback_more, _1, input_x, input_y));    
    //这里回调函数为三个参数时,使用boost::bind(&callback, _1, input_x, input_y),这里_1即为占位符,为subscriber接收到的/turtle1/pose_more话题中的内容占位;相当于callback(turtlesim::PoseConst& msg, input_x, input_y)
    
    ros::Rate loop_rate(1);
    int i=1;
    while(i<=3)
    {
        ros::spinOnce();
        loop_rate.sleep();
        i++;
    }
    return 0;
}

yocs_velocity_smoother.cpp

void VelocitySmoother::reconfigCB(yocs_velocity_smoother::paramsConfig &config, uint32_t level)
  {
    ROS_INFO("Reconfigure request : %f %f %f %f %f",
             config.speed_lim_v_x, config.speed_lim_v_y, config.speed_lim_w, config.accel_lim_v_x, config.accel_lim_v_y, config.accel_lim_w, config.decel_factor);

    speed_lim_v_x  = config.speed_lim_v_x;
    speed_lim_v_y  = config.speed_lim_v_y;
    speed_lim_w  = config.speed_lim_w;
    accel_lim_v_x  = config.accel_lim_v_x;
    accel_lim_v_y  = config.accel_lim_v_y;
    accel_lim_w  = config.accel_lim_w;
    decel_factor = config.decel_factor;
    decel_lim_v_x  = decel_factor*accel_lim_v_x;
    decel_lim_v_y  = decel_factor*accel_lim_v_y;
    decel_lim_w  = decel_factor*accel_lim_w;
  }

bool VelocitySmoother::init(ros::NodeHandle& nh)
  {
    // Dynamic Reconfigure 占位符_1, _2相当回调函数reconfigCB的config和level参数变量
    //相当于reconfigCB(yocs_velocity_smoother::paramsConfig &config, uint32_t level)
    dynamic_reconfigure_callback = boost::bind(&VelocitySmoother::reconfigCB, this, _1, _2);

    dynamic_reconfigure_server = new dynamic_reconfigure::Server<yocs_velocity_smoother::paramsConfig>(nh);
    dynamic_reconfigure_server->setCallback(dynamic_reconfigure_callback);
}

参考:https://blog.csdn.net/weixin_46181372/article/details/110951454

二.yocs_velocity_smoother.cpp速度平滑处理spin()函数理解

归一化确定最大速度增量

速度增量:\Delta v = v_{current} - v_{front}

A = (\Delta v,\Delta w) \gg (\frac{\left | \Delta v \right |}{\sqrt{\Delta v^{2}+\Delta w^{2}}}, \frac{\left | \Delta w\right |}{\sqrt{\Delta v^{2}+\Delta w^{2}}})

B = (\Delta v_{max},\Delta w_{max}) \gg (\frac{\left | \Delta v_{max} \right |}{\sqrt{\Delta v_{max}^{2}+\Delta w_{max}^{2}}}, \frac{\left | \Delta w_{max}\right |}{\sqrt{\Delta v_{max}^{2}+\Delta w_{max}^{2}}})

夹角

\theta = arctan(\frac{By}{Bx}) - arctan(\frac{Ay}{Ax})

如果\theta > 0

\frac{By}{Bx} - \frac{Ay}{Ax} > 0 \rightarrow \frac{By}{Bx} > \frac{Ay}{Ax} \overset{transfrom}{\rightarrow}By > \frac{Ay}{Ax} *Bx

最大线速度增量在以速度增量为基下时小于给定最大角度增量:

\Delta w_{max} = \frac{\left | \Delta w \right |}{\left | \Delta v \right |}*\Delta v_{max}

如果\theta < 0

\frac{By}{Bx} - \frac{Ay}{Ax} < 0 \rightarrow \frac{By}{Bx} < \frac{Ay}{Ax} \overset{transfrom}{\rightarrow}Bx > \frac{Ax}{Ay} *By

同理

\Delta v_{max} = \frac{\left | \Delta v \right |}{\left | \Delta w \right |}*\Delta w_{max}

输出速度:

\left | \Delta v \right | > \Delta v_{max}

v_{out} = v_{last} + sign(\Delta v) *\Delta v_{max}

\left | \Delta w \right | > \Delta w_{max}

w_{out} = w_{last} + sign(\Delta w) *\Delta w_{max}

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

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

相关文章

一篇文章搞懂:词法作用域、动态作用域、回调函数及闭包

前言 把以前一直只限于知道&#xff0c;却不清晰理解的这几个概念完完整整地梳理了一番。内容参考自wiki页面&#xff0c;然后加上自己一些理解。 词法作用域和动态作用域 不管什么语言&#xff0c;我们总要学习作用域(或生命周期)的概念&#xff0c;比如常见的称呼&#xf…

Flutter ーー logger 组件记录日志

Flutter ーー logger 组件记录日志 原文 https://medium.com/simbu/flutter-logging-with-logger-6227308ca199 前言 是时候添加一些日志记录了&#xff0c;我希望能够检查发生的网络请求和关键操作&#xff0c;在应用程序和后端之间的交互变得越来越复杂的情况下&#xff0c;给…

vue2自定义指令及钩子函数

目录​​​​​​​ 一、自定义指令的生命周期 二、局部自定义指令 三、全局自定义指令 1.定义自定义指令 2. index.js install方法安装自定义指令 3. main.js 全局挂载自定义指令 4.使用全局自定义指令 一、自定义指令的生命周期 在Vue中&#xff0c;自定义指令的生命…

晶振不仅仅是可以振荡就够了

晶振相当于人的心脏&#xff0c;能跳动&#xff0c;整个系统才是“活的”。晶振常见有有源晶振、无源晶振。有源晶振比较贵&#xff0c;但是需要外围电路少&#xff0c;供个电就能工作。无源晶振价格便宜&#xff0c;匹配电路复杂些。以无源晶振进行分析&#xff0c;以下是无源…

华为IdeaHub与华为云会议:软硬结合,天生一对

伴随数字化进程&#xff0c;传统视频会议终端存在着配置成本高昂、操作繁琐、组网复杂、需专业维护等问题&#xff0c;已无法满足企业数字化转型的需求&#xff0c;而会议软件使用便捷&#xff0c;但仍存在着与设备兼容性差、画面清晰度低、稳定性及信息安全等问题。面对市场上…

【疑难攻关】——文件包含漏洞

作者名&#xff1a;Demo不是emo 主页面链接&#xff1a;主页传送门 创作初心&#xff1a;舞台再大&#xff0c;你不上台&#xff0c;永远是观众&#xff0c;没人会关心你努不努力&#xff0c;摔的痛不痛&#xff0c;他们只会看你最后站在什么位置&#xff0c;然后羡慕或鄙夷座…

线性回归实现原理

一.定义 回归是监督学习的一个重要问题&#xff0c;回归用于预测输入变量和输出变量之间的关系&#xff0c;特别是当输入变量的值发生变化时&#xff0c;输出变量的值也随之发生变化。回归模型正是表示从输入变量到输出变量之间映射的函数 回归的目的是预测数组型的目标值。 …

并查集的学习

并查集是一种树型的数据结构&#xff0c;并查集可以高效地进行如下操作&#xff1a; 1、查询元素p和元素q是否属于同一组 2、合并元素p和元素q所在地组 并查集结构&#xff1a; 是一种树型结构&#xff0c;这棵树地要求比较简单 1、每个元素都唯一对应一个节点 2、每一组数…

iOS打包错误The operation couldn’t be completed. (AppThinning.StubError error 1.)

1、iOS打包错误 iOS打包报错&#xff1a;The operation couldn’t be completed. (AppThinning.StubError error 1.) 操作流程&#xff1a;archive之后选择Distribute App&#xff0c;在如上图步骤选择 Ad Hoc&#xff0c;然后出现The operation couldn’t be completed. (Ap…

GCC - 基于win10平台搭建Cmake + MinGW + gcc-arm-none 开源开发环境

前言 基于GUN开源工具链&#xff0c;搭建Windows平台下ARM编译运行环境&#xff01; &#x1f431;‍&#x1f680; 文中涉及的开发工具包已打包上传&#xff0c;可点击此处下载。 文章速览前言一、安装git-bash二、安装 MinGW-w6432位下载地址:  [MinGW - Minimalist GNU …

鉴源论坛 · 观模丨基于搜索的测试生成

作者 | 孙海英 华东师范大学软件工程学院讲师 苏亭 华东师范大学软件工程学院教授 版块 | 鉴源论坛 观模 测试用例自动生成&#xff0c;简称测试生成&#xff08;Test Generation&#xff09;&#xff0c;是指针对给定的被测对象&#xff0c;例如代码单元、接口、系统等&…

[附源码]Python计算机毕业设计Django防疫物资捐赠

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;我…

Spring MVC 中的分页 RESTful API 响应

分页允许您将来自Spring MVC的大型RESTful API响应拆分为称为页面的较小块。在这篇文章中&#xff0c;让我们看看如何使用Spring MVC和Spring JPA对来自Spring boot应用程序的JSON响应进行分页。 Spring MVC 中的分页和排序 如前所述&#xff0c;我们可以使用spring 数据 JPA…

急诊预检分诊管理系统的设计与实现

摘 要 随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的管理系统应运而生&#xff0c;各行各业相继进入信息管理时代&a…

webpack之性能优化

今天先来看两种性能优化的方式&#xff1a; 第一&#xff1a;抽离css代码并压缩 在没有配置之前&#xff0c;css是一起被打包到js文件里面的&#xff0c;像这样子 因此我们需要将css文件单独抽离出来 npm i mini-css-extract-plugin -D我们是需要在打包的时候去抽离它&#x…

LotuS2:新一代扩增子数据分析神器(更快、更准、更稳定)

一、研究背景 扩增子测序是分析微生物组成熟且成本较低的方法。然而&#xff0c;扩增子数据处理需要生物信息学技能和高计算能力来处理大数据集。此外&#xff0c;只有小部分工具适用于长读长扩增子数据分析。2022年10月Microbiome发表了一种新的扩增子数据处理神器——LotuS2…

0113 链表Day2

剑指 Offer 06. 从尾到头打印链表 输入一个链表的头节点&#xff0c;从尾到头反过来返回每个节点的值&#xff08;用数组返回&#xff09;。 示例 1 输入&#xff1a;head [1,3,2] 输出&#xff1a;[2,3,1] /*** Definition for singly-linked list.* public class ListN…

通过源码来理解Cglib与JDK动态代理

最近在阅读到了Spring源码对于两种动态代理使用在不同场景下的使用&#xff0c;两种方式各有利弊写一篇文加深自己的认识。文中对于源码的涉及较少&#xff0c;更多的是作者自己的理解和举例&#xff0c;然后通过部分源码验证。 首先看两个面试经常会遇到的关于Spring的问题&a…

纳尼?华为首席架构师只用434页笔记,就将网络协议给拿下了

不管是前端还是后端&#xff0c;几乎所有的程序运行都会涉及到网络协议。10 个程序员里面&#xff0c;10 个都说自己学过网络协议&#xff0c;9 个说自己懂网络协议。但真正面试的时候&#xff0c;能回答出相关问题的&#xff0c;可能只有两三个。 金九银十跳槽热季&#xff0…

七、【React-Router6】路由传参 之 search

文章目录1、routes.js2、Message.jsx3、Detail.jsx4、Result5、另外一个可以获取 search 的新 Hook &#xff1a;useLocation项目修改自 上一节 的 Demo 1、routes.js import { Navigate } from react-router-dom import About from ../components/About import Home from ../…