实体机器人识别虚拟环境中障碍物

之前的内容已经实现了虚拟机器人识别实体机器人的功能,接下来就是实体机器人如何识别虚拟环境中的障碍物(包括虚拟环境中的障碍物和其他虚拟机器人)。

我做的是基于雷达的,所以主要要处理的是雷达的scan话题

我的虚拟机器人命名空间在Vir_wheeltec_01下,实体机器人的命名空间在wheeltec_01下。

因此需要将虚拟机器人的雷达/Vir_wheeltec_01/scan 内容与/wheeltec_01/scan进行融合,由于虚拟机器人的雷达范围更大,因此融合的逻辑是以/wheeltec_01/scan作为掩模,把所有的障碍物都封存在新的雷达话题中/wheeltec_01/fused_scan,这一部分的功能封存在实体机器人的工作空间下的turn_on_wheeltec_robot功能包的scan_filter.py文件中

#!/usr/bin/env python
# -*- coding: utf-8 -*-

import rospy
import math
from sensor_msgs.msg import LaserScan
from message_filters import ApproximateTimeSynchronizer, Subscriber
from threading import Lock

class ScanFusionNode:
    def __init__(self):
        # 初始化发布器
        self.fused_scan_pub = rospy.Publisher('/wheeltec_01/fused_scan', LaserScan, queue_size=10)
        
        # 创建时间同步器
        real_sub = Subscriber('/wheeltec_01/scan', LaserScan)
        virtual_sub = Subscriber('/Vir_wheeltec_01/scan', LaserScan)
        
        self.ts = ApproximateTimeSynchronizer([real_sub, virtual_sub], queue_size=5, slop=0.05)
        self.ts.registerCallback(self.sync_callback)
        
        rospy.loginfo("scan_filter node initialized")

    def sync_callback(self, real_scan, virtual_scan):
        try:
            fused_scan = self.fuse_scans(real_scan, virtual_scan)
            self.fused_scan_pub.publish(fused_scan)
        except Exception as e:
            rospy.logerr("Fusion error: %s", str(e))

    def fuse_scans(self, real_scan, virtual_scan):
        # 参数校验
        if not self.validate_scans(real_scan, virtual_scan):
            raise ValueError("Invalid scan parameters")
        
        # 创建融合后的消息
        fused_scan = LaserScan()
        fused_scan.header = real_scan.header
        fused_scan.header.stamp = rospy.Time.now()
        fused_scan.angle_min = real_scan.angle_min
        fused_scan.angle_max = real_scan.angle_max
        fused_scan.angle_increment = real_scan.angle_increment
        fused_scan.time_increment = real_scan.time_increment
        fused_scan.scan_time = real_scan.scan_time
        fused_scan.range_min = real_scan.range_min
        fused_scan.range_max = real_scan.range_max
        
        # 计算虚拟扫描角度步长
        virtual_angle_step = (virtual_scan.angle_max - virtual_scan.angle_min) / len(virtual_scan.ranges)
        
        # 执行融合
        fused_ranges = []
        for i in range(len(real_scan.ranges)):
            # 获取实体数据
            real_range = real_scan.ranges[i]
            
            # 映射到虚拟扫描索引
            current_angle = real_scan.angle_min + i * real_scan.angle_increment
            virtual_idx = int((current_angle - virtual_scan.angle_min) / virtual_angle_step)
            virtual_idx = max(0, min(virtual_idx, len(virtual_scan.ranges)-1))
            virtual_range = virtual_scan.ranges[virtual_idx]
            
            # 有效性检查
            if math.isnan(real_range) or math.isinf(real_range):
                real_range = fused_scan.range_max
            if math.isnan(virtual_range) or math.isinf(virtual_range):
                virtual_range = fused_scan.range_max
                
            # 融合逻辑
            fused_range = min(real_range, virtual_range) if virtual_range <= fused_scan.range_max else real_range
            fused_ranges.append(fused_range)
        
        fused_scan.ranges = fused_ranges
        return fused_scan

    def validate_scans(self, real_scan, virtual_scan):
        # 校验角度范围是否重叠
        if (real_scan.angle_min > virtual_scan.angle_max) or (real_scan.angle_max < virtual_scan.angle_min):
            rospy.logwarn("Scan angle ranges do not overlap!")
            return False
        return True

if __name__ == '__main__':
    try:
        rospy.init_node('scan_filter')
        node = ScanFusionNode()
        rospy.spin()
    except rospy.ROSInterruptException:
        pass

接着就是基于这个话题去生成cost_map

找到costmap_car_params.yaml文件,修改如下(主要是添加fused_scan,因为小车中原来是没有这一段的,move_base默认就是基于scan话题生成costmap,所以要显式修改)

# 机器人外形设置参数,直接影响代价地图
footprint: [[-0.133, -0.125], [-0.133, 0.125], [0.133, 0.125], [0.133, -0.125]]

# 设置膨胀层参数,根据obstacle_layer、static_layer和footprint生成代价地图
inflation_layer:
  enabled: true  # 是否开启膨胀层
  cost_scaling_factor: 15  # 代价地图数值随与障碍物距离下降的比值,越大会导致路径规划越靠近障碍物
  inflation_radius: 0.25  # 机器人膨胀半径,影响路径规划,单位:m

# 激光扫描传感器配置,确保配置正确的缩进
observation_sources: laser_scan_sensor
laser_scan_sensor:
  data_type: LaserScan
  topic: /wheeltec_01/fused_scan
  marking: true
  clearing: true

 由于修改后move_base会基于fused_scan生成cosmap所以要在movebase节点启动前就生成fused_scan雷达信息还需要修改实体小车的启动文件wxfnavigation.launch(可以teb补全,因为具体的我忘记了),主要就是在启动雷达后加入(scan_filter.py要做成可执行文件 chomd +x 文件路径/文件名)

“<!-- 启动扫描数据融合节点 scan_filter.py -->
    <node name="scan_filter" pkg="your_package_name" type="scan_filter.py" output="screen">
      <remap from="scan" to="/wheeltec_01/scan" />
      <remap from="fused_scan" to="/wheeltec_01/fused_scan" />
    </node>”

<launch>
  <!-- 使用命名空间 wheeltec_01 -->
  <group ns="wheeltec_01">
    
    <!-- 开启机器人底层相关节点 同时开启导航功能 -->
    <include file="$(find turn_on_wheeltec_robot)/launch/turn_on_wheeltec_robot.launch">
      <arg name="navigation" default="true"/>
    </include>

    <!-- turn on lidar 开启雷达 -->
    <include file="$(find turn_on_wheeltec_robot)/launch/wheeltec_lidar.launch" />

    <!-- 启动扫描数据融合节点 scan_filter.py -->
    <node name="scan_filter" pkg="your_package_name" type="scan_filter.py" output="screen">
      <remap from="scan" to="/wheeltec_01/scan" />
      <remap from="fused_scan" to="/wheeltec_01/fused_scan" />
    </node>

    <!-- 设置需要用于导航的地图 -->
    <arg name="map_file" default="$(find turn_on_wheeltec_robot)/map/WHEELTEC.yaml"/>
    <node name="map_server_for_test" pkg="map_server" type="map_server" args="$(arg map_file)"/>

    <!-- 开启用于导航的自适应蒙特卡洛定位 amcl -->
    <include file="$(find turn_on_wheeltec_robot)/launch/include/amcl.launch" >
    
    <node pkg="tf" type="static_transform_publisher" name="wheeltec_world_to_global_01" args="-9 -4 0 0 0 0 world /map 100" />

    <!-- MarkerArray功能节点 -->
    <node name='send_mark' pkg="turn_on_wheeltec_robot" type="send_mark.py"/>

    <!-- move_base 节点检测 -->
    <node name='node_detection' pkg="turn_on_wheeltec_robot" type="node_ping.py">
      <param name="node_name" type="string" value="/move_base"/>
    </node>

  </group> <!-- 结束命名空间 wheeltec_01 -->
</launch>

并且启动时,需要先在主机启动虚拟小车(gazebo)roslaunch wxfpublish testSimulation.launch

接着启动小车的导航文件roslaunch turn_on ... wxfnavigation.launch(可以teb补全,因为具体的我忘记了)

最终结果

此外地图文件更换成map,并且修改实体小车下map中的位置,把rviz中实体小车到world的映射改成0000就可以了

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

湖北中医药大学谱度众合(武汉)生命科技有限公司研究生工作站揭牌

2025年2月11日&#xff0c;湖北中医药大学&谱度众合&#xff08;武汉&#xff09;生命科技有限公司研究生工作站揭牌仪式在武汉生物技术研究院一楼101会议室举行&#xff0c;湖北中医药大学研究生院院长刘娅教授、基础医学院院长孔明望教授、基础医学院赵敏教授、基础医学院…

ARM Coretex-M核心单片机(STM32)找到hardfault的原因,与hardfault解决方法

1. 前提基础知识&#xff08;ARM异常 压栈流程&#xff09;M核栈增长是地址逐渐减小的 **M3h ARM CM4核心带浮点处理器FPU的&#xff0c;压栈的东西还不一样 进入hardfult后看MSP或者SP的值&#xff0c;看下边第二章图如果hardfult里边啥都没有&#xff0c;就只有个while(1){}…

组件传递props校验

注意&#xff1a;prop是只读的&#xff01;不可以修改父组件的数据。 可以检验传过来的内容是否类型没问题。 App.vue <template><div><!-- <parentDemo/> --><componentA/></div></template> <script> import ComponentA …

机试刷题_NC52 有效括号序列【python】

NC52 有效括号序列 from operator import truediv # # 代码中的类名、方法名、参数名已经指定&#xff0c;请勿修改&#xff0c;直接返回方法规定的值即可 # # # param s string字符串 # return bool布尔型 # class Solution:def isValid(self , s: str) -> bool:if not s…

threejs:document.createElement创建标签后css设置失效

vue3threejs&#xff0c;做一个给模型批量CSS2D标签的案例&#xff0c;在导入模型的js文件里&#xff0c;跟着课程写的代码如下&#xff1a; import * as THREE from three; // 引入gltf模型加载库GLTFLoader.js import { GLTFLoader } from three/addons/loaders/GLTFLoader.…

一文读懂西门子 PLC 串口转以太网系列模块

在工业自动化领域&#xff0c;随着智能化和信息化的不断发展&#xff0c;设备之间的高效通信变得至关重要。西门子 PLC 作为工业控制的核心设备&#xff0c;其通信方式的拓展需求日益凸显。西门子 PLC 串口转网口产品应运而生&#xff0c;它为实现串口设备与以太网网络的无缝连…

Linux | GRUB / bootloader 详解

注&#xff1a;本文为 “Linux | GRUB / bootloader” 相关文章合辑。 英文引文&#xff0c;机翻未校。 图片清晰度限于引文原状。 未整理去重。 What is Grub in Linux? What is it Used for? Linux 中的 Grub 是什么&#xff1f;它的用途是什么&#xff1f; Abhishek …

java高级(IO流多线程)

file 递归 字符集 编码 乱码gbk&#xff0c;a我m&#xff0c;utf-8 缓冲流 冒泡排序 //冒泡排序 public static void bubbleSort(int[] arr) {int n arr.length;for (int i 0; i < n - 1; i) { // 外层循环控制排序轮数for (int j 0; j < n -i - 1; j) { // 内层循环…

Dubbo RPC 原理

一、Dubbo 简介 Apache Dubbo 是一款高性能、轻量级的开源 RPC 框架&#xff0c;支持服务治理、协议扩展、负载均衡、容错机制等核心功能&#xff0c;广泛应用于微服务架构。其核心目标是解决分布式服务之间的高效通信与服务治理问题。 二、Dubbo 架构设计 1. 核心组件 Prov…

普中单片机-51TFT-LCD显示屏(1.8寸 STM32)

普中官方论坛&#xff1a; http://www.prechin.cn/gongsixinwen/208.html 普中科技-各型号开发板资料链接&#xff1a;https://www.bilibili.com/read/cv23681775/?spm_id_from333.999.0.0 27-TFTLCD显示实验_哔哩哔哩_bilibili 2.程序烧录 2.1设置彩屏驱动 3.实验效果

Starlink卫星动力学系统仿真建模第九讲-滑模(SMC)控制算法原理简介及卫星控制应用

滑模控制&#xff08;Sliding Mode Control&#xff09;算法详解 一、基本原理 滑模控制&#xff08;Sliding Mode Control, SMC&#xff09;是一种变结构控制方法&#xff0c;通过设计一个滑模面&#xff08;Sliding Surface&#xff09;&#xff0c;迫使系统状态在有限时间内…

nss刷题5(misc)

[HUBUCTF 2022 新生赛]最简单的misc 打开后是一张图片&#xff0c;没有其他东西&#xff0c;分离不出来&#xff0c;看看lsb&#xff0c;红绿蓝都是0&#xff0c;看到头是png&#xff0c;重新保存为png&#xff0c;得到一张二维码 扫码得到flag [羊城杯 2021]签到题 是个动图…

【C/C++】删除链表的倒数第 N 个结点(leetcode T19)

考点预览&#xff1a; 双指针法&#xff1a;通过维护两个指针来一次遍历链表&#xff0c;避免了多次遍历链表的低效方法。 边界条件&#xff1a;要特别处理删除头结点的情况。 题目描述&#xff1a; 给你一个链表&#xff0c;删除链表的倒数第 n 个结点&#xff0c;并且返回…

人工智能定义

一、人工智能核心概念体系 1.1 人工智能的本质 人工智能的定义:人工智能(Artificial Intelligence,简称 AI)是指计算机系统能够执行通常需要人类智能才能完成的任务,如学习、推理、解决问题、理解自然语言、识别图像和声音等。它通过模拟人类的智能行为,运用算法和数据…

量子计算的威胁,以及企业可以采取的措施

当谷歌、IBM、Honeywell和微软等科技巨头纷纷投身量子计算领域时&#xff0c;一场技术军备竞赛已然拉开帷幕。 量子计算虽能为全球数字经济带来巨大价值&#xff0c;但也有可能对相互关联的系统、设备和数据造成损害。这一潜在影响在全球网络安全领域引起了强烈关注。也正因如…

0—QT ui界面一览

2025.2.26&#xff0c;感谢gpt4 1.控件盒子 1. Layouts&#xff08;布局&#xff09; 布局控件用于组织界面上的控件&#xff0c;确保它们的位置和排列方式合理。 Vertical Layout&#xff08;垂直布局&#xff09; &#xff1a;将控件按垂直方向排列。 建议&#xff1a;适…

【Uniapp-Vue3】导入uni-id用户体系

在uniapp官网的uniCloud中下载uni-id用户体系 或者直接进入加载&#xff0c;下载地址&#xff1a;uni-id-pages - DCloud 插件市场 进入以后下载插件&#xff0c;打开HbuilderX 选中项目&#xff0c;点击确定 点击跳过 点击合并 右键uniCloud文件夹下的database文件夹&#x…

如何免费使用稳定的deepseek

0、背景&#xff1a; 在AI辅助工作中&#xff0c;除了使用cursor做编程外&#xff0c;使用deepseek R1进行问题分析、数据分析、代码分析效果非常好。现在我经常会去拿行业信息、遇到的问题等去咨询R1&#xff0c;也给了自己不少启示。但是由于官网稳定性很差&#xff0c;很多…

VSCode+PlatformIO报错 找不到头文件

如图示&#xff0c;找不到目标头文件 demo工程运行正常&#xff0c;考虑在src文件夹内开辟自己的代码&#xff0c;添加后没有找到 找了些资料&#xff0c;大概记录如下&#xff1a; 1、c_cpp_properties.json 内记录 头文件配置 .vscode 中&#xff0c;此文件是自动生成的&a…

Python 网络爬虫实战全解析:案例驱动的技术探索

Python 网络爬虫实战全解析&#xff1a;案例驱动的技术探索 本文围绕 Python 网络爬虫展开&#xff0c;深入剖析其技术要点&#xff0c;并通过实际案例演示开发流程。从爬虫原理引入&#xff0c;逐步讲解如何使用 Python 中的requests和BeautifulSoup等库进行网页数据抓取与解…