ros2激光雷达<gazebo>仿真资料

Lidar sensor激光雷达传感器

We don't want our robot to touch the wall at all because this may cause some damage, so instead of the contact sensor we can use the Lidar. Lidar is an acronym for "light detection and ranging". This sensor can help us detect obstacles around the robot. We will use it to measure the distance between our robot and the wall.

First let's create a frame to fix our lidar to. This should be added inside of the vehicle_blue <model> tag, since the lidar frame is attached to the robot's chassis:

<frame name="lidar_frame" attached_to='chassis'><pose>0.8 0 0.5 0 0 0</pose>
</frame>

Then add this plugin under the <world> tag, to be able to use the lidar sensor:

    <pluginfilename="gz-sim-sensors-system"name="gz::sim::systems::Sensors"><render_engine>ogre2</render_engine></plugin>

Under the chassis link we can add the lidar sensor as follows:

<sensor name='gpu_lidar' type='gpu_lidar'>"<pose relative_to='lidar_frame'>0 0 0 0 0 0</pose><topic>lidar</topic><update_rate>10</update_rate><ray><scan><horizontal><samples>640</samples><resolution>1</resolution><min_angle>-1.396263</min_angle><max_angle>1.396263</max_angle></horizontal><vertical><samples>1</samples><resolution>0.01</resolution><min_angle>0</min_angle><max_angle>0</max_angle></vertical></scan><range><min>0.08</min><max>10.0</max><resolution>0.01</resolution></range></ray><always_on>1</always_on><visualize>true</visualize>
</sensor>
  • First we defined the name and type of our sensor, then we defined its <pose> relative to the lidar_frame.
  • In the <topic> we define the topic on which the lidar data will be published.
  • <update_rate> is the frequency at which the lidar data is generated, in our case 10 Hz which is equal to 0.1 sec.
  • Under the <horizontal> and <vertical> tags we define the properties of the horizontal and vertical laser rays.
  • <samples> is the number of simulated lidar rays to generate per complete laser sweep cycle.
  • <resolution>: this number is multiplied by samples to determine the number range data points.
  • The <min_angle> and <max_angle> are the angle range of the generated rays.
  • Under the <range> we define range properties of each simulated ray
    • <min> and <max> define the minimum and maximum distance for each lidar ray.
    • The <resolution> tag here defines the linear resolution of each lidar ray.
  • <always_on>: if true the sensor will always be updated according to the <update_rate>.
  • <visualize>: if true the sensor is visualized in the GUI.

Now run the world and press the play button in the bottom-left corner:

gz sim sensor_tutorial.sdf

Look at the lidar messages on the /lidar topic, specifically the ranges data:

gz topic -e -t /lidar

The lidar message has the following attributes:

message LaserScan
{Header header              = 1;string frame               = 2;Pose world_pose            = 3;double angle_min           = 4;double angle_max           = 5;double angle_step          = 6;double range_min           = 7;double range_max           = 8;uint32 count               = 9;double vertical_angle_min  = 10;double vertical_angle_max  = 11;double vertical_angle_step = 12;uint32 vertical_count      = 13;repeated double ranges              = 14;repeated double intensities         = 15;
}

Avoid the wall

Now as we have the lidar on our robot, we can use the ranges data and make our robot avoid hitting the wall. To do that, we'll write a short C++ program that listens to the sensor data and sends velocity commands to the robot. This program is called a node. We will build a node that subscribes to the /lidar topic and reads its data. Have a look at this tutorial to learn how to build a publisher and a subscriber node. You can download the finished node for this demo from here.

The lidar_node
gz::transport::Node node;
std::string topic_pub = "/cmd_vel";
gz::msgs::Twist data;
auto pub = node.Advertise<gz::msgs::Twist>(topic_pub);

We declare a node which will publish to cmd_vel topic and defined the message type Twist. Then advertise our node.

void cb(const gz::msgs::LaserScan &_msg)
{bool allMore = true;for (int i = 0; i < _msg.ranges_size(); i++){if (_msg.ranges(i) < 1.0){allMore = false;break;}}if (allMore) //if all bigger than one{data.mutable_linear()->set_x(0.5);data.mutable_angular()->set_z(0.0);}else{data.mutable_linear()->set_x(0.0);data.mutable_angular()->set_z(0.5);}pub.Publish(data);
}

Inside the callback function we check if the range of all rays are bigger than 1.0. If so we publish a message to our car to move forward. Otherwise we make the robot rotate.

int main(int argc, char **argv)
{std::string topic = "/lidar";// Subscribe to a topic by registering a callback.if (!node.Subscribe(topic, cb)){std::cerr << "Error subscribing to topic [" << topic << "]" << std::endl;return -1;}// Zzzzzz.gz::transport::waitForShutdown();return 0;
}

Inside the main we subscribe to the lidar topic, and wait until the node is shut down.

Build the node

Download the CMakeLists.txt, and in the same folder of lidar_node create build/ directory:

mkdir build
cd build

Run cmake and build the code:

cmake ..
make lidar_node
Run the node

Run the node from terminal 1:

./build/lidar_node

Run the world from terminal 2:

gz sim sensor_tutorial.sdf

Now you can see the robot move forward and as it approaches the wall it starts to turn left until it's clear and moves forward again (be sure to press the play button in the bottom-left corner to make the robot start moving).

Gazebo launch

Instead of running two different tasks from two different terminals we can make a launch file which will run the sensor_world and the lidar_node at the same time. Open your text editor and add the following code.

<?xml version='1.0'?>
<gz version='1.0'><executable name='sensor-world'><command>gz sim sensor_tutorial.sdf</command></executable><executable name='lidar_node'><command>./build/lidar_node</command></executable></gz>

The launch file is an XML file. We simply define what commands will run under the <executable> tag. The first command is gz sim sensor_tutorial.sdf which launches the world. And the second command is ./build/lidar_node which runs the lidar_node. Save the file as sensor_launch.gzlaunch, and then run it using the following command:

gz launch sensor_launch.gzlaunch

Press the play button to start the simulation. Hurray! Our robot is now moving and avoiding the wall.

To add even more complexity to your simulation, learn how to add actors to a world in the next tutorial.

Video walk-through

A video walk-through of this tutorial is available from our YouTube channel: Gazebo tutorials: Sensors

/

<!--ros2示例代码-->
<!--https://github.com/ros-simulation/gazebo_ros_pkgs/wiki/ROS-2-Migration:-Ray-sensors-->
<link name="hokuyo_link"><!-- Visuals / Collisions omitted for this example --><sensor type="gpu_ray" name="head_hokuyo_sensor"><ray><scan><horizontal><samples>720</samples><resolution>1</resolution><min_angle>-1.570796</min_angle><max_angle>1.570796</max_angle></horizontal></scan><range><min>0.10</min><max>30.0</max><resolution>0.01</resolution></range><!-- Using gazebo's noise instead of plugin's --><noise><type>gaussian</type><mean>0.0</mean><stddev>0.01</stddev></noise></ray><!-- Using gazebo's update rate instead of plugin's --><update_rate>30</update_rate><plugin name="gazebo_ros_head_hokuyo_controller" filename="libgazebo_ros_ray_sensor.so"><!-- Change namespace and output topic so published topic is /rrbot/laser/scan --><ros><namespace>/rrbot/laser</namespace><remapping>~/out:=scan</remapping></ros><!-- Set output to sensor_msgs/LaserScan to get same output type as gazebo_ros_laser --><output_type>sensor_msgs/LaserScan</output_type><!-- <frame_name> ommited, will default to hokuo_link --></plugin></sensor>
</link>

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

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

相关文章

OpenFeign相关面试题及答案(2024)

1、什么是OpenFeign&#xff0c;它如何简化远程服务调用&#xff1f; OpenFeign是一个声明式的Web服务客户端&#xff0c;它使得编写HTTP客户端变得更加容易。它属于Spring Cloud Netflix项目的一部分&#xff0c;可以与Spring Boot应用轻松集成。通过使用OpenFeign&#xff0…

Lingo 17安装包下载及安装教程

Lingo 17下载链接&#xff1a;https://docs.qq.com/doc/DUndEVXd4WVVweGFR 1.鼠标右键解压到“Lingo 17.0” 2.双击打开【Setup】文件夹 3.选中Lingo 17.0&#xff0c;鼠标右键选择“以管理员身份运行” 4.点击“Next” 5.选中I accept the terms in the license agreement&…

go语言`json:“-“`标签的含义

json:"-" 是 Go 语言中的一个标签&#xff08;tag&#xff09;&#xff0c;用于指示编码和解码 JSON 时忽略对应的字段。 在 Go 中&#xff0c;结构体的字段可以通过添加标签来指定其在编码为 JSON 字符串或解码时的行为。json:"-" 标签的作用是告诉编码和…

第四篇 行为型设计模式 - 灵活定义对象间交互

第四篇&#xff1a;行为型设计模式 - 灵活定义对象间交互 行为型设计模式关注对象之间的交互和职责分配&#xff0c;旨在定义对象间的高效、灵活的通信机制。以下是十一种常见行为型设计模式的详解及其应用场景。 1. 策略模式详解及其应用场景 详解&#xff1a; 策略模式定义…

贪心算法part05 435无重叠区间

435无重叠区间 763 划分字母区间 56合并区间

javascript中location对象的属性与方法

前言 本章介绍js中的location中的属性和方法。 文章目录 前言什么是location为什么要用locationlocation对象属性location对象方法总结 什么是location 在JavaScript中&#xff0c;location 是一个包含当前页面的URL信息的对象。它允许你获取和操作当前页面的URL&#xff0c;比…

速通C语言第十二站 文件操作

系列文章目录 速通C语言系列 速通C语言第一站 一篇博客带你初识C语言 http://t.csdn.cn/N57xl 速通C语言第二站 一篇博客带你搞定分支循环 http://t.csdn.cn/Uwn7W 速通C语言第三站 一篇博客带你搞定函数 http://t.csdn.cn/bfrUM 速通C语言第四站 一篇博客带…

QML 项目中使用 Qt Design Studio 生成的UI界面

作者&#xff1a;billy 版权声明&#xff1a;著作权归作者所有&#xff0c;商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处 前言 今天来和大家聊一下 Qt Design Studio 这个软件。这个软件的主要功能是用来快速完成 UI 界面&#xff0c;就和 widget 中的 desig…

Vue.js 中使用 Watch 选项实现动态问题判断与展示答案

组件结构 以下是组件的基本结构&#xff1a; <template><div><!-- 输入框&#xff0c;用于输入问题 --><p>提出一个是/否问题&#xff1a;<input v-model"question" :disabled"loading" /></p><!-- 显示答案 --&…

栈实现后缀表达式的计算

后缀表达式计算 过程分析 中缀表达式 &#xff08;15&#xff09;*3 > 后缀表达式 153* (可参考这篇文章&#xff1a;中缀转后缀) 第一步&#xff1a;我们从左至右扫描 后缀表达式(已经存放在一个字符数组中)&#xff0c;遇到第一个数字字符 ‘1’ 放入栈中第二步&#xf…

市场复盘总结 20240103

仅用于记录当天的市场情况,用于统计交易策略的适用情况,以便程序回测 短线核心:不参与任何级别的调整 昨日回顾: 方法一:指标选股 select * from dbo.ResultAll where 入选类型 like %指标选股% and 入选日期=20240103;方法二:趋势选股法 1、最低价持续3日上涨 2、均价…

新的一年,新的征程,35岁,再出发!!

2024&#xff0c;迈入新的征程&#xff01;35岁是人生的一个重要阶段&#xff0c;是积累经验、提升自我、迎接挑战的黄金时期&#xff0c;在接下来的日子&#xff1a; 设定目标&#xff1a;明确自己的长期和短期目标&#xff0c;确保自己始终朝着正确的方向前进。目标可以是职…

jmeter线程组

特点&#xff1a;模拟用户&#xff0c;支持多用户操作&#xff1b;可以串行也可以并行 分类&#xff1a; setup线程组&#xff1a;初始化 类似于 unittest中的setupclass 普通线程组&#xff1a;字面意思 teardown线程组&#xff1a;环境恢复&#xff0c;后置处理

机器视觉系统选型-选型-总结

一&#xff1a;明确需求 需求&#xff1a;镜面材质上的划痕检测&#xff0c;传送线上运动过程中拍照&#xff0c;无景深要求&#xff0c;传送线速度0.8m/s&#xff0c;产品间隔50mm 产品大小&#xff1a;100*80mm 工作距离限制&#xff1a;≤ 300mm 最小划痕宽度&#xff1a;0.…

shell编程二

shell 脚本规范 shell脚本文件需要以.sh结尾 第一个原因&#xff0c;让别人认的这个是shell脚本&#xff0c;sh后缀编辑时有高亮显示。 拓展名后缀,如果省略.sh则不易判断该文件是否为shell脚本 ​ # 执行脚本方式 1、 sh 脚本.sh 2、 bash 脚本.sh 3、 ./脚本.sh # 需要执行权…

NVMe SSD IO压力导致宕机案例解读-2

IOVA原理扩展介绍&#xff1a; 在Linux内核的I/O虚拟地址&#xff08;IOVA&#xff09;分配机制中存在两个影响高吞吐量I/O性能的问题 问题1&#xff1a;原有IOVA分配器在分配时可能需要对已分配的IOVA范围进行线性搜索&#xff0c;这种操作效率低下&#xff0c;尤其在大规模…

Day23

Day23 一,file类 1.1file类的初识 import java.io.File; import java.text.SimpleDateFormat;public class Test01 {/*** 知识点&#xff1a;File类* File&#xff0c;是文件和目录路径名的抽象表示* File只关注文件本身的信息&#xff0c;而不能操作文件里面的内容* * File…

AIGC时代-GPT-4和DALL·E 3的结合

在当今这个快速发展的数字时代&#xff0c;人工智能&#xff08;AI&#xff09;已经成为了我们生活中不可或缺的一部分。从简单的自动化任务到复杂的决策制定&#xff0c;AI的应用范围日益扩大。而在这个广阔的领域中&#xff0c;有两个特别引人注目的名字&#xff1a;GPT-4和D…

webpack 5 mode的作用和区别

通过选择 development, production 或 none 之中的一个&#xff0c;来设置 mode 参数&#xff0c;你可以启用 webpack 内置在相应环境下的优化。其默认值为 production。 会将 DefinePlugin 中 process.env.NODE_ENV 的值设置为 developmen或者production. 为模块和 chunk 启用…

JUC原子操作类

原子操作类 基本类型原子类&#xff1a;AtomicInteger、AtomicBoolean、AtomicLong&#xff0c;常见API&#xff1a; get 获取当前值getAndSet 获取当前的值&#xff0c;并设置新的值getAndIncrement 获取当前的值&#xff0c;并自增getAndDecrement 获取当前的值&#xff0c;并…