跳到主要内容
极客日志极客日志面向AI+效率的开发者社区
首页博客GitHub 精选镜像工具UI配色美学隐私政策关于联系
搜索内容 / 工具 / 仓库 / 镜像...⌘K搜索
注册
博客列表
PythonAI算法

PX4 无人机 MID360 使用 FAST_LIO 实现自主飞行及定点配置

介绍基于 PX4 固件、MID360 激光雷达及 FAST_LIO 算法的无人机室内自主定位与定点悬停配置流程。涵盖硬件选型(MicoAir743V2 飞控、Ubuntu 机载电脑)、PX4 参数设置(视觉定位源、关闭罗盘)、ROS 节点开发(Fast-LIO 到 MAVROS 位姿转换脚本)及坐标系验证。重点说明雷达频率对精度的影响及实机运动测试方法,为室内无人机组建提供技术参考。

虚拟内存发布于 2026/4/6更新于 2026/5/1420 浏览
PX4 无人机 MID360 使用 FAST_LIO 实现自主飞行及定点配置

PX4 固件版本为 1.15.4

QGC 地面站版本为 4.4.5

飞控:微空科技 MicoAir743V2

机载电脑:12 代 i5, Ubuntu 20.04

安装位置:MID360 的接口对应飞机的后方

一、PX4 飞控设置

1. 将 PX4 定位数据源设置为 Vision

参数 EKF2_EV_CTRL:可以默认,或者看参数手册按需配置(该教程默认)。

参数 EKF2_HGT_REF:Vision

2. 关闭罗盘

需要注意的是,你现在走完上面飞控的设置,机头上电后会指向北。但在走完下面的程序后,你会发现上电后无人机机头指向东方(罗盘已经被关了)。

此外,加一个激光测距融合好处多多。

二、程序

注意,下面我把雷达等频率改到了 30HZ,但是在这频率下,定位精度实测只能有 10hz 的 60% 不到,慎重选择是否改频率!!!!!

1. 代码

接下来创建发布 /mavros/vision_pose/pose 话题的功能包过程了。这里只提供代码,具体创建工作空间和功能包的步骤建议查阅官方文档或相关教程。主要功能:将转化后的位姿信息以话题 /mavros/vision_pose/pose 发布。

#!/usr/bin/env python3
import rospy
from nav_msgs.msg import Odometry
from geometry_msgs.msg import PoseStamped
import tf
import numpy as np
from collections import deque
import math

# 滑动窗口平均类,用于平滑 yaw 值
class SlidingWindowAverage:
    def __init__(self, window_size):
        self.window_size = window_size
        self.data_queue = deque()
        self.window_sum = 0.0

    def add_data(self, new_data):
        if self.data_queue and abs(new_data - .data_queue[-]) > :
            .data_queue.clear()
            .window_sum = 
        .data_queue.append(new_data)
        .window_sum += new_data
         (.data_queue) > .window_size:
            .window_sum -= .data_queue.popleft()
         .window_sum / (.data_queue)

     ():
         (.data_queue)

     ():
         .data_queue:
             .window_sum / (.data_queue)
        :
             

 :
     ():
        rospy.init_node(, anonymous=)
        .p_lidar_body = np.zeros()
        .q_mav = [, , , ]
        .q_px4_odom = [, , , ]
        .window_size = 
        .swa = SlidingWindowAverage(.window_size)
        .init_flag = 
        .init_q = tf.transformations.quaternion_from_euler(, , )
        rospy.Subscriber(, Odometry, .vins_callback)
        rospy.Subscriber(, Odometry, .px4_odom_callback)
        .vision_pub = rospy.Publisher(, PoseStamped, queue_size=)
        .rate = rospy.Rate()
        .run()

     ():
        euler = tf.transformations.euler_from_quaternion(q)
         euler[]

     ():
        .p_lidar_body = np.array([
            msg.pose.pose.position.x,
            msg.pose.pose.position.y,
            msg.pose.pose.position.z
        ])
        .q_mav = [
            msg.pose.pose.orientation.x,
            msg.pose.pose.orientation.y,
            msg.pose.pose.orientation.z,
            msg.pose.pose.orientation.w
        ]

     ():
        .q_px4_odom = [
            msg.pose.pose.orientation.x,
            msg.pose.pose.orientation.y,
            msg.pose.pose.orientation.z,
            msg.pose.pose.orientation.w
        ]
        yaw = .from_quaternion_to_yaw(.q_px4_odom)
        .swa.add_data(yaw)

     ():
          rospy.is_shutdown():
             .swa.get_size() == .window_size   .init_flag:
                init_yaw = .swa.get_avg()
                .init_q = tf.transformations.quaternion_from_euler(, , init_yaw)
                .init_flag = 
             .init_flag:
                rot_matrix = tf.transformations.quaternion_matrix(.init_q)[:, :]
                p_enu = np.dot(rot_matrix, .p_lidar_body)
                vision = PoseStamped()
                vision.header.stamp = rospy.Time.now()
                vision.header.frame_id = 
                vision.pose.position.x = p_enu[]
                vision.pose.position.y = p_enu[]
                vision.pose.position.z = p_enu[]
                vision.pose.orientation.x = .q_mav[]
                vision.pose.orientation.y = .q_mav[]
                vision.pose.orientation.z = .q_mav[]
                vision.pose.orientation.w = .q_mav[]
                .vision_pub.publish(vision)
                rospy.loginfo(.(
                    p_enu[], p_enu[], p_enu[], .q_mav[], .q_mav[], .q_mav[], .q_mav[]))
                .rate.sleep()

 __name__ == :
    :
        FastLIOToMavros()
     rospy.ROSInterruptException:
        
self
1
0.01
self
self
0.0
self
self
if
len
self
self
self
self
return
self
len
self
def
get_size
self
return
len
self
def
get_avg
self
if
self
return
self
len
self
else
return
0.0
class
FastLIOToMavros
def
__init__
self
'fastlio_to_mavros'
True
self
3
self
0
0
0
1
self
0
0
0
1
self
8
self
self
self
False
self
0
0
0
'/Odometry'
self
'/mavros/local_position/odom'
self
self
'/mavros/vision_pose/pose'
10
self
30.0
self
def
from_quaternion_to_yaw
self, q
return
2
def
vins_callback
self, msg
self
self
def
px4_odom_callback
self, msg
self
self
self
self
def
run
self
while
not
if
self
self
and
not
self
self
self
0
0
self
True
if
self
self
3
3
self
"map"
0
1
2
self
0
self
1
self
2
self
3
self
"\nPosition in ENU:\n x: {:.3f}\n y: {:.3f}\n z: {:.3f}\nOrientation of LiDAR:\n x: {:.3f}\n y: {:.3f}\n z: {:.3f}\n w: {:.3f}"
format
0
1
2
self
0
self
1
self
2
self
3
self
if
'__main__'
try
except
pass

自行创建 fastlio_to_mavros 节点的 launch 文件。

2. 运行 MID360 和 FAST_LIO 的程序,修改雷达扫描频率

按照官方说法修改 livox_ros_driver2 msg_MID360.launch,使其频率达到 30HZ(其实默认也能用)。方法在 livox_ros_driver2 的 GitHub 底下写了。

文章配图

文章配图

roslaunch livox_ros_driver2 msg_MID360.launch

在另一个终端中执行:

roslaunch fast_lio mapping_mid360.launch

输入下图指令查看是否修改了。

文章配图

3. 运行 MAVROS

roslaunch mavros px4.launch

必须运行上面指令后马上运行下面的 fastlio_to_mavros。

4. 运行自己创建的 fastlio_to_mavros 节点的 launch 文件

运行结果,和 QGC 里的画面(飞控通过数传和 Windows 电脑 QGC 连接,故而频率很低)。

文章配图

三、验证

推荐用官方的仿真来理解坐标系。

你会发现一运行 fastlio_to_mavros 节点,无人机机头突然从指向北变成指向东方(确保罗盘已经被关了)。

在 Ubuntu 上位机看输入 rostopic echo /mavros/local_position/pose 机头所指的方向为正 X。X 在前,Y 朝向左,Z 朝向下。

打开 QGC、点左上角 Analyze Tools——>MAVlink 检测,出现了 LOCAL_POSITION_NED 数据,坐标系看法如下:NED 坐标系,X 为北,Y 为东,Z 为下,机头指向东,那么向东运动(机头方向),Y 会增大。向北运动(飞机左方向),X 会增大。那么向上运动,Z 会是负数,且越来越负。

实机向前方运动

会发现 QGC 的 LOCAL_POSITION_NED 的 Y 增大,/mavros/local_position/pose 的 X 增大。

文章配图

实机向右方运动

会发现 QGC 的 LOCAL_POSITION_NED 的 X 变小为负数,/mavros/local_position/pose 的 Y 变小为负数。

文章配图

当飞机向上运动

会发现 QGC 的 LOCAL_POSITION_NED 的 Z 变成负数并且不断减小。

文章配图

四、总结

实机建议调好 PID 再起飞,仅试过定点飞行,无异常问题。

目录

  1. 一、PX4 飞控设置
  2. 1. 将 PX4 定位数据源设置为 Vision
  3. 2. 关闭罗盘
  4. 二、程序
  5. 1. 代码
  6. 滑动窗口平均类,用于平滑 yaw 值
  7. 2. 运行 MID360 和 FAST_LIO 的程序,修改雷达扫描频率
  8. 3. 运行 MAVROS
  9. 4. 运行自己创建的 fastliotomavros 节点的 launch 文件
  10. 三、验证
  11. 实机向前方运动
  12. 实机向右方运动
  13. 当飞机向上运动
  14. 四、总结
  • 💰 8折买阿里云服务器限时8折了解详情
  • Magick API 一键接入全球大模型注册送1000万token查看
  • 🤖 一键搭建Deepseek满血版了解详情
  • 一键打造专属AI 智能体了解详情
极客日志微信公众号二维码

微信扫一扫,关注极客日志

微信公众号「极客日志V2」,在微信中扫描左侧二维码关注。展示文案:极客日志V2 zeeklog

更多推荐文章

查看全部
  • Spring Cloud Gateway 微服务统一入口与核心配置
  • CTF 竞赛 Web 题目基本解题流程
  • Docker Compose 部署 OpenClaw 并接入飞书机器人
  • MATLAB Compiler SDK 中的 mwArray 核心类使用详解
  • C++11 详解:列表初始化、右值引用与移动语义
  • voidImageViewer:支持 GIF/WEBP 动画的轻量级图像查看器
  • C++ 类与对象内在机制:运算符重载、赋值重载与取址重载
  • 人工智能大模型训练过程与自监督学习原理解析
  • 大模型算法面试指南:核心问题与答案解析
  • StreamVLN 具身导航复现指南:环境搭建与模型推理
  • GitHub Copilot 及主流 AI 编程模型对比与选型指南
  • 知网 AIGC 检测不通过的处理思路与步骤
  • SecureCRT 下载与安装及基础配置指南
  • 前端无障碍性开发指南:构建包容性 Web 体验
  • Ubuntu 24.04 LTS 虚拟机安装与基础环境配置实战
  • OpenAI Whisper 语音转文字完全使用指南
  • DeepSeek 各版本演进与优缺点深度解析
  • Java 连接 SQL Server 的两种 JDBC 实现方式
  • LLaMA-Factory:大语言模型预训练、微调、评估与部署实战指南
  • 42 个 Python 实用小例子

相关免费在线工具

  • 加密/解密文本

    使用加密算法(如AES、TripleDES、Rabbit或RC4)加密和解密文本明文。 在线工具,加密/解密文本在线工具,online

  • RSA密钥对生成器

    生成新的随机RSA私钥和公钥pem证书。 在线工具,RSA密钥对生成器在线工具,online

  • Mermaid 预览与可视化编辑

    基于 Mermaid.js 实时预览流程图、时序图等图表,支持源码编辑与即时渲染。 在线工具,Mermaid 预览与可视化编辑在线工具,online

  • 随机西班牙地址生成器

    随机生成西班牙地址(支持马德里、加泰罗尼亚、安达卢西亚、瓦伦西亚筛选),支持数量快捷选择、显示全部与下载。 在线工具,随机西班牙地址生成器在线工具,online

  • Gemini 图片去水印

    基于开源反向 Alpha 混合算法去除 Gemini/Nano Banana 图片水印,支持批量处理与下载。 在线工具,Gemini 图片去水印在线工具,online

  • curl 转代码

    解析常见 curl 参数并生成 fetch、axios、PHP curl 或 Python requests 示例代码。 在线工具,curl 转代码在线工具,online