ROS机器人Diego 1#制作(二十二)基于EAI F4激光雷达数据进行定位amcl

8 篇文章 1 订阅
7 篇文章 0 订阅

更多创客作品,请关注笔者网站园丁鸟,搜集全球极具创意,且有价值的创客作品


在上两篇博文中我们讲到了利用gmapping和hector生成室内地图,本文将阐述如何利用激光雷达数据,在已产生的地图上进行定位。
1.生成地图
创建maps文件夹,并赋予其777的权限,既可以让map_server在此文件夹中生成地图文件
这里写图片描述

打开终端cd进入maps目录,并执行地图生成命令

cd ~/catkin_ws/src/diego_nav/maps/
rosrun map_server map_saver -f f4_gmapping

执行完后,在maps目录下将生成两个地图文件
这里写图片描述
地图效果如下
这里写图片描述

2.配置定位的launch文件

<launch>
  <master auto="start"/>

  <include file="$(find flashgo)/launch/lidar.launch" />
  
  <node name="arduino" pkg="ros_arduino_python" type="arduino_node.py" output="screen">
      <rosparam file="$(find ros_arduino_python)/config/my_arduino_params.yaml" command="load" />
  </node>  
   
  <node pkg="tf" type="static_transform_publisher" name="base_frame_2_laser_link" args="0.0 0.0 0.2 3.14 3.14 0 /base_link /laser 40"/>   
  
  <!-- Map server -->
  <node name="map_server" pkg="map_server" type="map_server" args="$(find diego_nav)/maps/f4_gmapping_whole1.yaml" /> 

  <!-- amcl node -->
  <node pkg="amcl" type="amcl" name="amcl" output="screen">
  <remap from="scan" to="scan"/>
  <!-- Publish scans from best pose at a max of 10 Hz -->
  <param name="initial_pose_x" value="0.0"/>
  <param name="initial_pose_y" value="0.0"/>
  <param name="initial_pose_a" value="0.0"/>
  <param name="use_map_topic" value="true"/>
  <param name="odom_model_type" value="diff"/>
  <param name="odom_alpha5" value="0.1"/>
  <param name="transform_tolerance" value="0.5" />
  <param name="gui_publish_rate" value="10.0"/>
  <param name="laser_max_beams" value="300"/>
  <param name="min_particles" value="500"/>
  <param name="max_particles" value="5000"/>
  <param name="kld_err" value="0.1"/>
  <param name="kld_z" value="0.99"/>
  <param name="odom_alpha1" value="0.1"/>
  <param name="odom_alpha2" value="0.1"/>
  <!-- translation std dev, m -->
  <param name="odom_alpha3" value="0.1"/>
  <param name="odom_alpha4" value="0.1"/>
  <param name="laser_z_hit" value="0.9"/>
  <param name="laser_z_short" value="0.05"/>
  <param name="laser_z_max" value="0.05"/>
  <param name="laser_z_rand" value="0.5"/>
  <param name="laser_sigma_hit" value="0.2"/>
  <param name="laser_lambda_short" value="0.1"/>
  <param name="laser_lambda_short" value="0.1"/>
  <param name="laser_model_type" value="likelihood_field"/>
  <!-- <param name="laser_model_type" value="beam"/> -->
  <param name="laser_min_range" value="1"/>
  <param name="laser_max_range" value="8"/>
  <param name="laser_likelihood_max_dist" value="2.0"/>
  <param name="update_min_d" value="0.2"/>
  <param name="update_min_a" value="0.5"/>
  <param name="resample_interval" value="1"/>
  <param name="transform_tolerance" value="0.1"/>
  <param name="recovery_alpha_slow" value="0.0"/>
  <param name="recovery_alpha_fast" value="0.0"/>


  </node>

</launch>

这里需要强调的一点是必须配置机器人的其实位置点 ,机器人随便放一个位置,amcl并不会定位到,需要指定初始位置,地图的原点是用gmapping/hector构建地图时候的起始点,我们在这里就把机器人放在起始的位置,所以初始位置的值都设定为0.0

 <param name="initial_pose_x" value="0.0"/>
 <param name="initial_pose_y" value="0.0"/>

3.启动launch文件

roslaunch diego_nav diego_run_gmapping_amcl_flashgo.launch

打开新的终端,启动键盘控制节点

rosrun teleop_twist_keyboard teleop_twist_keyboard.py

这时我们可以用键盘控制机器人动起来,并打开rviz查看定位情况

rosin rviz rviz

这里写图片描述
这张图中我们可以看到一堆红色的箭头暨蒙特卡洛定位算法所产生的粒子分布,其中箭头方向是机器人运动的方向
这里写图片描述
这张图中可以看粒子比上张图更加聚合,从实际观察来看聚合度高的情况下定位效果更加。

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
使用IMU和激光雷达进行机器人定位,可以采用扩展卡尔曼滤波(Extended Kalman Filter,EKF)算法。 首先,需要在ROS环境中安装`robot_localization`功能包,可以通过以下命令进行安装: ``` sudo apt-get install ros-<distro>-robot-localization ``` 其中,`<distro>`是ROS版本号,例如`kinetic`、`melodic`等。 接着,在ROS节点中引入`robot_localization`的头文件,并创建一个`ros::NodeHandle`对象: ``` #include <ros/ros.h> #include <robot_localization/ekf_localization_node.hpp> ... ros::NodeHandle nh("~"); ``` 然后,需要设置EKF的参数,例如状态量、传感器数据类型等: ``` std::vector<std::string> state_vars = {"x", "y", "z", "roll", "pitch", "yaw", "xd", "yd", "zd", "rolld", "pitchd", "yawd"}; // 状态量 std::vector<std::string> odom_vars = {"x", "y", "z", "roll", "pitch", "yaw"}; // 里程计数据 std::vector<std::string> imu_vars = {"roll", "pitch", "yaw", "rolld", "pitchd", "yawd"}; // IMU数据 std::vector<std::string> laser_vars = {"x", "y", "z"}; // 激光雷达数据 robot_localization::EkfLocalizationNode::EkfConfig config; config.set_state_vars(state_vars); config.set_odom_vars(odom_vars); config.set_imu_vars(imu_vars); config.set_laser_vars(laser_vars); ``` 接着,可以通过以下代码创建EKF节点: ``` robot_localization::EkfLocalizationNode ekf_node(config); ekf_node.setNodeHandle(&nh); ekf_node.init(); ``` 最后,可以在`ros::spin()`循环中调用EKF节点的定位函数,例如: ``` while (ros::ok()) { ekf_node.correct(); // 使用IMU和激光雷达数据进行校正 pose = ekf_node.getRobotPose(); // 获取机器人位姿 ... ros::spinOnce(); } ``` 这样就可以使用C++编写ROS机器人使用IMU和激光雷达进行定位了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

DiegoRobot

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值