news 2026/6/10 10:34:40

无人机高精度悬停:GPS+光流EKF融合方案

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
无人机高精度悬停:GPS+光流EKF融合方案

目录

1. 融合原理

GPS 与光流的特性对比

融合目标

2. EKF 设计(二维位置融合)

状态向量

状态方程(预测)

观测方程(更新)

GPS 观测矩阵:

光流观测矩阵:

3. 代码实现(STM32 HAL 库)

4. 集成到无人机悬停控制

数据读取与融合流程

5. 参数调优与测试建议

参数初始值

调优步骤


我们在之前的位置 + 姿态双环 PID基础上,增加GPS + 光流融合的扩展卡尔曼滤波(EKF),用于高精度位置估计,从而提升无人机悬停精度到±0.5 米以内。

我会给你一个完整的融合方案,包括:

  1. 融合原理(GPS 与光流的互补性)
  2. EKF 状态方程与观测方程设计
  3. 代码实现(STM32 HAL 库)
  4. 参数调优与测试建议

1. 融合原理

GPS 与光流的特性对比

传感器优势劣势
GPS全局绝对位置,无累计误差噪声大(±2~5 米),刷新率低(1~10Hz)
光流高刷新率(50~200Hz),相对位移精度高有累计误差,依赖高度信息,无全局定位

融合目标

  • 短期:用光流的高刷新率数据保持位置跟踪稳定性。
  • 长期:用 GPS 的绝对位置修正光流的累计误差。
  • 结果:输出一个低噪声、高刷新率、无累计误差的位置估计。

2. EKF 设计(二维位置融合)

状态向量

假设我们只关心水平位置(X, Y),状态向量为:X=​xyvx​vy​​​其中:

  • x,y:位置(米)
  • vx​,vy​:速度(米 / 秒)

状态方程(预测)

基于匀速模型:Xk∣k−1​=F⋅Xk−1∣k−1​+wk​状态转移矩阵:F=​1000​0100​dt010​0dt01​​过程噪声协方差矩阵 Q:Q=​σx2​000​0σy2​00​00σvx​2​0​000σvy​2​​​

观测方程(更新)

有两个观测源:

  1. GPS:直接观测位置 (xgps​,ygps​)
  2. 光流:观测位移 (Δxflow​,Δyflow​),需转换为速度或位置增量
GPS 观测矩阵:

Hgps​=[10​01​00​00​]观测噪声协方差矩阵 Rgps​:Rgps​=[σgps,x2​0​0σgps,y2​​]

光流观测矩阵:

光流输出位移 Δxflow​,Δyflow​,假设采样时间为 dt,则速度观测为:vflow,x​=dtΔxflow​​,vflow,y​=dtΔyflow​​观测矩阵:Hflow​=[00​00​10​01​]观测噪声协方差矩阵 Rflow​:Rflow​=[σflow,x2​0​0σflow,y2​​]


3. 代码实现(STM32 HAL 库)

#include <math.h> // EKF 状态向量 typedef struct { float x, y; // 位置 (m) float vx, vy; // 速度 (m/s) } EKF_State; // EKF 协方差矩阵 typedef struct { float P[4][4]; } EKF_Covariance; // 传感器数据 typedef struct { float x, y; // GPS 位置 (m) } GPS_Data; typedef struct { float dx, dy; // 光流位移 (m) } Flow_Data; // EKF 参数 typedef struct { EKF_State state; EKF_Covariance cov; float dt; // 采样时间 (s) // 过程噪声 float Q_pos; // 位置过程噪声方差 float Q_vel; // 速度过程噪声方差 // 观测噪声 float R_gps_pos; // GPS 位置噪声方差 float R_flow_vel; // 光流速度噪声方差 } EKF; // 初始化 EKF void EKF_Init(EKF *ekf, float dt) { ekf->dt = dt; ekf->Q_pos = 0.01f; // 位置过程噪声 ekf->Q_vel = 0.1f; // 速度过程噪声 ekf->R_gps_pos = 4.0f; // GPS 位置噪声方差 (2m^2) ekf->R_flow_vel = 0.01f; // 光流速度噪声方差 (0.1m/s)^2 // 初始状态 ekf->state.x = 0.0f; ekf->state.y = 0.0f; ekf->state.vx = 0.0f; ekf->state.vy = 0.0f; // 初始协方差 for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { ekf->cov.P[i][j] = 0.0f; } } ekf->cov.P[0][0] = 1.0f; // 初始位置不确定度 ekf->cov.P[1][1] = 1.0f; ekf->cov.P[2][2] = 1.0f; // 初始速度不确定度 ekf->cov.P[3][3] = 1.0f; } // 预测步骤 void EKF_Predict(EKF *ekf) { // 状态预测 ekf->state.x += ekf->state.vx * ekf->dt; ekf->state.y += ekf->state.vy * ekf->dt; // 协方差预测 float F[4][4] = { {1, 0, ekf->dt, 0}, {0, 1, 0, ekf->dt}, {0, 0, 1, 0}, {0, 0, 0, 1} }; float Q[4][4] = { {ekf->Q_pos, 0, 0, 0}, {0, ekf->Q_pos, 0, 0}, {0, 0, ekf->Q_vel, 0}, {0, 0, 0, ekf->Q_vel} }; // P = F*P*F^T + Q float P_temp[4][4]; for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { P_temp[i][j] = 0.0f; for (int k = 0; k < 4; k++) { P_temp[i][j] += F[i][k] * ekf->cov.P[k][j]; } } } for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { ekf->cov.P[i][j] = 0.0f; for (int k = 0; k < 4; k++) { ekf->cov.P[i][j] += P_temp[i][k] * F[j][k]; } ekf->cov.P[i][j] += Q[i][j]; } } } // 更新步骤(GPS) void EKF_Update_GPS(EKF *ekf, GPS_Data *gps) { // 观测矩阵 H = [1,0,0,0; 0,1,0,0] float H[2][4] = { {1, 0, 0, 0}, {0, 1, 0, 0} }; // 观测噪声 R float R[2][2] = { {ekf->R_gps_pos, 0}, {0, ekf->R_gps_pos} }; // 残差 z - H*x float z[2] = {gps->x, gps->y}; float z_pred[2] = {ekf->state.x, ekf->state.y}; float y[2]; y[0] = z[0] - z_pred[0]; y[1] = z[1] - z_pred[1]; // S = H*P*H^T + R float S[2][2]; float H_P[2][4]; for (int i = 0; i < 2; i++) { for (int j = 0; j < 4; j++) { H_P[i][j] = 0.0f; for (int k = 0; k < 4; k++) { H_P[i][j] += H[i][k] * ekf->cov.P[k][j]; } } } for (int i = 0; i < 2; i++) { for (int j = 0; j < 2; j++) { S[i][j] = 0.0f; for (int k = 0; k < 4; k++) { S[i][j] += H_P[i][k] * H[j][k]; } S[i][j] += R[i][j]; } } // 卡尔曼增益 K = P*H^T*S^{-1} float S_inv[2][2]; float det = S[0][0] * S[1][1] - S[0][1] * S[1][0]; S_inv[0][0] = S[1][1] / det; S_inv[0][1] = -S[0][1] / det; S_inv[1][0] = -S[1][0] / det; S_inv[1][1] = S[0][0] / det; float K[4][2]; for (int i = 0; i < 4; i++) { for (int j = 0; j < 2; j++) { K[i][j] = 0.0f; for (int k = 0; k < 4; k++) { K[i][j] += ekf->cov.P[i][k] * H[j][k]; } } } for (int i = 0; i < 4; i++) { for (int j = 0; j < 2; j++) { float temp = 0.0f; for (int k = 0; k < 2; k++) { temp += K[i][k] * S_inv[k][j]; } K[i][j] = temp; } } // 更新状态 for (int i = 0; i < 4; i++) { float delta = 0.0f; for (int j = 0; j < 2; j++) { delta += K[i][j] * y[j]; } if (i == 0) ekf->state.x += delta; else if (i == 1) ekf->state.y += delta; else if (i == 2) ekf->state.vx += delta; else if (i == 3) ekf->state.vy += delta; } // 更新协方差 P = (I - K*H)*P float I_KH[4][4]; for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { I_KH[i][j] = (i == j) ? 1.0f : 0.0f; for (int k = 0; k < 2; k++) { I_KH[i][j] -= K[i][k] * H[k][j]; } } } float P_temp[4][4]; for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { P_temp[i][j] = 0.0f; for (int k = 0; k < 4; k++) { P_temp[i][j] += I_KH[i][k] * ekf->cov.P[k][j]; } } } for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { ekf->cov.P[i][j] = P_temp[i][j]; } } } // 更新步骤(光流) void EKF_Update_Flow(EKF *ekf, Flow_Data *flow) { // 光流位移转速度 float vx_flow = flow->dx / ekf->dt; float vy_flow = flow->dy / ekf->dt; // 观测矩阵 H = [0,0,1,0; 0,0,0,1] float H[2][4] = { {0, 0, 1, 0}, {0, 0, 0, 1} }; // 观测噪声 R float R[2][2] = { {ekf->R_flow_vel, 0}, {0, ekf->R_flow_vel} }; // 残差 z - H*x float z[2] = {vx_flow, vy_flow}; float z_pred[2] = {ekf->state.vx, ekf->state.vy}; float y[2]; y[0] = z[0] - z_pred[0]; y[1] = z[1] - z_pred[1]; // S = H*P*H^T + R float S[2][2]; float H_P[2][4]; for (int i = 0; i < 2; i++) { for (int j = 0; j < 4; j++) { H_P[i][j] = 0.0f; for (int k = 0; k < 4; k++) { H_P[i][j] += H[i][k] * ekf->cov.P[k][j]; } } } for (int i = 0; i < 2; i++) { for (int j = 0; j < 2; j++) { S[i][j] = 0.0f; for (int k = 0; k < 4; k++) { S[i][j] += H_P[i][k] * H[j][k]; } S[i][j] += R[i][j]; } } // 卡尔曼增益 K = P*H^T*S^{-1} float S_inv[2][2]; float det = S[0][0] * S[1][1] - S[0][1] * S[1][0]; S_inv[0][0] = S[1][1] / det; S_inv[0][1] = -S[0][1] / det; S_inv[1][0] = -S[1][0] / det; S_inv[1][1] = S[0][0] / det; float K[4][2]; for (int i = 0; i < 4; i++) { for (int j = 0; j < 2; j++) { K[i][j] = 0.0f; for (int k = 0; k < 4; k++) { K[i][j] += ekf->cov.P[i][k] * H[j][k]; } } } for (int i = 0; i < 4; i++) { for (int j = 0; j < 2; j++) { float temp = 0.0f; for (int k = 0; k < 2; k++) { temp += K[i][k] * S_inv[k][j]; } K[i][j] = temp; } } // 更新状态 for (int i = 0; i < 4; i++) { float delta = 0.0f; for (int j = 0; j < 2; j++) { delta += K[i][j] * y[j]; } if (i == 0) ekf->state.x += delta; else if (i == 1) ekf->state.y += delta; else if (i == 2) ekf->state.vx += delta; else if (i == 3) ekf->state.vy += delta; } // 更新协方差 P = (I - K*H)*P float I_KH[4][4]; for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { I_KH[i][j] = (i == j) ? 1.0f : 0.0f; for (int k = 0; k < 2; k++) { I_KH[i][j] -= K[i][k] * H[k][j]; } } } float P_temp[4][4]; for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { P_temp[i][j] = 0.0f; for (int k = 0; k < 4; k++) { P_temp[i][j] += I_KH[i][k] * ekf->cov.P[k][j]; } } } for (int i = 0; i < 4; i++) { for (int j = 0; j < 4; j++) { ekf->cov.P[i][j] = P_temp[i][j]; } } }

4. 集成到无人机悬停控制

数据读取与融合流程

EKF ekf; GPS_Data gps; Flow_Data flow; void Hover_Control_Loop(void) { float dt = 0.01f; // 100Hz // 1. 读取传感器数据 read_gps(&gps); // GPS 位置 (m) read_optical_flow(&flow); // 光流位移 (m) // 2. EKF 预测 EKF_Predict(&ekf); // 3. EKF 更新(GPS 每秒一次,光流每次循环) static uint32_t gps_update_count = 0; if (gps_update_count++ >= 100) { // 100Hz * 0.01s = 1s EKF_Update_GPS(&ekf, &gps); gps_update_count = 0; } EKF_Update_Flow(&ekf, &flow); // 4. 位置环 PID(使用 EKF 输出) pid_pos_x.setpoint = target_pos.x; pid_pos_x.feedback = ekf.state.x; PID_Update(&pid_pos_x, dt); pid_pos_y.setpoint = target_pos.y; pid_pos_y.feedback = ekf.state.y; PID_Update(&pid_pos_y, dt); // 5. 姿态环与电机控制(同上) ... }

5. 参数调优与测试建议

参数初始值

  • 过程噪声
    • Q_pos = 0.01:位置过程噪声较小,假设无人机运动平稳。
    • Q_vel = 0.1:速度过程噪声较大,允许速度变化。
  • 观测噪声
    • R_gps_pos = 4.0:GPS 噪声方差(2m)。
    • R_flow_vel = 0.01:光流速度噪声方差(0.1m/s)。

调优步骤

  1. 室内测试:关闭 GPS 更新,仅用光流,调整QR_flow使位置跟踪稳定。
  2. 室外测试:开启 GPS 更新,调整R_gps使 GPS 修正不过于频繁或剧烈。
  3. 最终验证:悬停时位置误差应控制在±0.5 米以内。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/23 0:55:58

无人机三维精准悬停:EKF融合GPS与气压计

目录 1. 高度融合原理 气压计与 GPS 高度特性对比 融合目标 2. 高度融合 EKF 设计 状态向量 状态方程&#xff08;预测&#xff09; 观测方程&#xff08;更新&#xff09; 观测矩阵&#xff1a; 观测噪声协方差矩阵&#xff1a; 3. 代码实现&#xff08;STM32 HAL 库…

作者头像 李华
网站建设 2026/5/22 3:06:36

校园迎新大学生新生入学报到系统 微信小程序的设计与实现_49tlsixa

目录已开发项目效果实现截图关于博主开发技术介绍核心代码参考示例1.建立用户稀疏矩阵&#xff0c;用于用户相似度计算【相似度矩阵】2.计算目标用户与其他用户的相似度系统测试总结源码文档获取/同行可拿货,招校园代理 &#xff1a;文章底部获取博主联系方式&#xff01;已开发…

作者头像 李华
网站建设 2026/6/8 18:37:25

YOLOv8 GhostNet极轻量替代选项探索

YOLOv8 GhostNet极轻量替代选项探索 在智能安防摄像头、工业巡检终端和消费级无人机等边缘设备上部署目标检测模型&#xff0c;常常面临一个尴尬的现实&#xff1a;算法精度达标了&#xff0c;但推理速度却卡在个位数帧率&#xff1b;模型结构优化了&#xff0c;可一跑torchsum…

作者头像 李华
网站建设 2026/6/9 10:42:20

YOLOv8雾天、雨天等恶劣天气适应性测试

YOLOv8在雾天、雨天等恶劣天气下的适应性实测与工程实践 在智能交通系统&#xff08;ITS&#xff09;和自动驾驶技术飞速发展的今天&#xff0c;视觉感知的鲁棒性已成为决定系统能否真正落地的关键瓶颈。摄像头作为最直观的传感器&#xff0c;其采集的数据极易受到雾、雨、低光…

作者头像 李华
网站建设 2026/5/23 2:34:55

青少年近视怎么预防?家长必知的核心要点你了解吗?

当下青少年近视发生率逐年攀升&#xff0c;已然成为困扰众多家庭的健康难题&#xff0c;不少家长重视孩子视力保护&#xff0c;却因缺乏专业认知&#xff0c;陷入防控误区&#xff0c;导致预防效果不尽如人意。青少年视力发育尚未成熟&#xff0c;眼部调节系统仍在完善&#xf…

作者头像 李华