无人机双环PID悬停控制全解析

无人机双环PID悬停控制全解析

目录

1. 无人机悬停控制系统架构

2. 位置 PID 控制器设计

PID 结构体(扩展到位置控制)

3. 位置控制与姿态控制结合

位置数据结构

位置环 PID 初始化

4. 位置控制循环

5. 完整控制流程(位置 + 姿态 + 电机)

6. 传感器数据融合(GPS / 光流)

GPS 数据读取示例

光流数据读取示例

7. 调试与优化建议


我们在之前的 姿态控制 PID 基础上,增加 位置控制 PID 层,这样无人机就可以根据 GPS 或 光流模块 提供的位置信息,实现精准悬停。

我会给你一个 完整的位置 + 姿态双环 PID 控制方案,包括:

  1. 系统架构(位置环 → 姿态环 → 电机混控)
  2. 位置 PID 设计
  3. 传感器数据融合(GPS / 光流 → 位置估计)
  4. 代码实现(STM32 HAL 库)
  5. 调试与优化建议

1. 无人机悬停控制系统架构

悬停控制是一个 双环 PID 结构

┌─────────────┐ ┌─────────────┐ ┌─────────────┐ ┌─────────────┐ │ 位置环 PID │ │ 姿态环 PID │ │ 角速度环PID │ │ 电机混控 │ │ (X/Y/Z) │ │ (Roll/Pitch/Yaw) │ │ (ωx/ωy/ωz) │ │ (Motor1~4) │ └──────┬──────┘ └──────┬──────┘ └──────┬──────┘ └──────┬──────┘ │ │ │ │ ▼ ▼ ▼ ▼ ┌─────────────┐ ┌─────────────┐ ┌─────────────┐ ┌─────────────┐ │ 期望位置 │ │ 期望姿态角 │ │ 期望角速度 │ │ PWM信号输出 │ └─────────────┘ └─────────────┘ └─────────────┘ └─────────────┘ 
  • 位置环:输入期望位置(如悬停点)和实际位置(来自 GPS / 光流),输出期望姿态角(Roll/Pitch)和期望高度(油门修正)。
  • 姿态环:输入期望姿态角和实际姿态角(来自 MPU-9250),输出期望角速度。
  • 角速度环:输入期望角速度和实际角速度(来自陀螺仪),输出电机控制量。
  • 电机混控:结合油门和姿态控制量,计算四个电机的 PWM 值。

2. 位置 PID 控制器设计

位置环需要对 X、Y、Z 三个方向分别控制:

  • X/Y 轴:控制无人机在水平面上的位置,输出期望的横滚角(Roll)和俯仰角(Pitch)。
  • Z 轴:控制无人机的高度,输出油门修正量。

PID 结构体(扩展到位置控制)

typedef struct { float Kp, Ki, Kd; float setpoint; float feedback; float error; float integral; float derivative; float prev_error; float output; float integral_limit; float output_limit; } PID_Controller; void PID_Init(PID_Controller *pid, float Kp, float Ki, float Kd, float integral_limit, float output_limit) { pid->Kp = Kp; pid->Ki = Ki; pid->Kd = Kd; pid->integral_limit = integral_limit; pid->output_limit = output_limit; pid->setpoint = 0.0f; pid->feedback = 0.0f; pid->error = 0.0f; pid->integral = 0.0f; pid->derivative = 0.0f; pid->prev_error = 0.0f; pid->output = 0.0f; } void PID_Update(PID_Controller *pid, float dt) { pid->error = pid->setpoint - pid->feedback; float P = pid->Kp * pid->error; pid->integral += pid->error * dt; if (pid->integral > pid->integral_limit) pid->integral = pid->integral_limit; if (pid->integral < -pid->integral_limit) pid->integral = -pid->integral_limit; float I = pid->Ki * pid->integral; pid->derivative = (pid->error - pid->prev_error) / dt; float D = pid->Kd * pid->derivative; pid->output = P + I + D; if (pid->output > pid->output_limit) pid->output = pid->output_limit; if (pid->output < -pid->output_limit) pid->output = -pid->output_limit; pid->prev_error = pid->error; } 

3. 位置控制与姿态控制结合

位置数据结构

typedef struct { float x; float y; float z; } Position; Position target_pos; // 期望位置 Position current_pos; // 当前位置(来自GPS/光流) 

位置环 PID 初始化

PID_Controller pid_pos_x, pid_pos_y, pid_pos_z; void Position_PID_Init(void) { // X轴位置PID(输出期望Roll角) PID_Init(&pid_pos_x, 2.0f, 0.1f, 0.05f, 50.0f, 15.0f); // 输出限幅±15° // Y轴位置PID(输出期望Pitch角) PID_Init(&pid_pos_y, 2.0f, 0.1f, 0.05f, 50.0f, 15.0f); // Z轴位置PID(输出油门修正量) PID_Init(&pid_pos_z, 5.0f, 0.2f, 0.1f, 100.0f, 200.0f); // 输出限幅±200us } 

4. 位置控制循环

void Position_Control_Loop(float dt) { // 更新位置环PID pid_pos_x.setpoint = target_pos.x; pid_pos_x.feedback = current_pos.x; PID_Update(&pid_pos_x, dt); pid_pos_y.setpoint = target_pos.y; pid_pos_y.feedback = current_pos.y; PID_Update(&pid_pos_y, dt); pid_pos_z.setpoint = target_pos.z; pid_pos_z.feedback = current_pos.z; PID_Update(&pid_pos_z, dt); // 位置环输出作为姿态环输入 pid_roll.setpoint = pid_pos_y.output; // Y轴位置误差→Roll角 pid_pitch.setpoint = pid_pos_x.output; // X轴位置误差→Pitch角 pid_yaw.setpoint = 0.0f; // 悬停时航向角保持0° // 油门修正 throttle = 1500.0f + pid_pos_z.output; // 基础油门1500us + Z轴修正 throttle = (throttle < 1000) ? 1000 : (throttle > 2000) ? 2000 : throttle; } 

5. 完整控制流程(位置 + 姿态 + 电机)

void Drone_Hover_Loop(void) { float dt = 0.01f; // 100Hz控制频率 // 1. 读取传感器数据 read_mpu9250_data(&euler, gyro); // 姿态角和角速度 read_position_sensor(&current_pos); // GPS/光流位置 // 2. 位置环控制 Position_Control_Loop(dt); // 3. 姿态环控制 pid_roll.feedback = euler.roll; pid_pitch.feedback = euler.pitch; pid_yaw.feedback = euler.yaw; PID_Update(&pid_roll, dt); PID_Update(&pid_pitch, dt); PID_Update(&pid_yaw, dt); // 4. 角速度环控制 pid_rate_roll.setpoint = pid_roll.output; pid_rate_pitch.setpoint = pid_pitch.output; pid_rate_yaw.setpoint = pid_yaw.output; pid_rate_roll.feedback = gyro[0]; pid_rate_pitch.feedback = gyro[1]; pid_rate_yaw.feedback = gyro[2]; PID_Update(&pid_rate_roll, dt); PID_Update(&pid_rate_pitch, dt); PID_Update(&pid_rate_yaw, dt); // 5. 电机混控 motor1 = throttle + pid_rate_roll.output - pid_rate_pitch.output - pid_rate_yaw.output; motor2 = throttle - pid_rate_roll.output - pid_rate_pitch.output + pid_rate_yaw.output; motor3 = throttle - pid_rate_roll.output + pid_rate_pitch.output - pid_rate_yaw.output; motor4 = throttle + pid_rate_roll.output + pid_rate_pitch.output + pid_rate_yaw.output; // 限幅 motor1 = (motor1 < 1000) ? 1000 : (motor1 > 2000) ? 2000 : motor1; motor2 = (motor2 < 1000) ? 1000 : (motor2 > 2000) ? 2000 : motor2; motor3 = (motor3 < 1000) ? 1000 : (motor3 > 2000) ? 2000 : motor3; motor4 = (motor4 < 1000) ? 1000 : (motor4 > 2000) ? 2000 : motor4; // 6. 输出到电机 __HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_1, motor1); __HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_2, motor2); __HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_3, motor3); __HAL_TIM_SET_COMPARE(&htim1, TIM_CHANNEL_4, motor4); } 

6. 传感器数据融合(GPS / 光流)

GPS 数据读取示例

void read_gps_data(Position *pos) { // 假设GPS模块通过串口输出NMEA数据 // 解析$GPGGA或$GPRMC帧,提取经纬度和高度 // 转换为局部坐标系(如ENU坐标系) pos->x = gps_lon_to_x(gps_lon); pos->y = gps_lat_to_y(gps_lat); pos->z = gps_alt; } 

光流数据读取示例

void read_optical_flow_data(Position *pos) { // 读取光流模块的X/Y位移和高度数据 pos->x += optical_flow_x * dt; pos->y += optical_flow_y * dt; pos->z = sonar_alt; // 超声波或气压计高度 } 

7. 调试与优化建议

  1. 参数整定顺序
    • 先调姿态环,再调位置环。
    • 位置环的 Kp 从小开始,逐步增加,避免振荡。
  2. 传感器数据滤波
    • GPS 数据噪声大,建议使用卡尔曼滤波或滑动平均。
    • 光流数据需结合高度信息进行缩放。
  3. 安全保护
    • 加入位置超范围保护(如偏离悬停点过远时自动返航)。
    • 低电压保护,避免失控。
  4. 仿真验证
    • 先用 MATLAB/Simulink 搭建仿真模型,验证位置控制逻辑。
  5. 实际测试
    • 先在室内用光流 + 超声波测试悬停,再到室外用 GPS 测试。

✅ 我已经给你一个 完整的位置 + 姿态双环 PID 悬停控制方案,可以直接在 STM32 上实现,支持 GPS 或光流模块。

如果你需要的话,我可以帮你 加上卡尔曼滤波(EKF)融合 GPS 和光流数据,这样位置估计会更稳定,尤其在 GPS 信号弱的环境下。

Read more

目前好用的AI Idea工具

目前好用的AI Idea工具

📃个人主页:编程的一拳超人 ⛺️ 欢迎关注:👍点赞 👂🏽留言 😍收藏 💞 💞 💞 于高山之巅,方见大河奔涌;于群峰之上,更觉长风浩荡。 ——《人民日报》 快速总览:按IDE生态与场景优先推荐——JetBrains IDE用户首选JetBrains AI Assistant(原生深度集成);VS Code用户优先GitHub Copilot X(生态成熟);国产免费优选文心快码/通义灵码;隐私优先选本地模型+JetBrains AI/CodeLlama;云原生场景用Amazon CodeWhisperer。以下是详细选型与能力对比。 文章目录 * 一、主流IDE原生AI能力(推荐首选) * 1. JetBrains AI Assistant(IntelliJ IDEA等专用) * 2. GitHub Copilot X(VS Code/IntelliJ等通用) * 3.

Flutter 组件 google_generative_language_api 适配鸿蒙 HarmonyOS 实战:生成式 AI 集成,构建大语言模型调度与全场景智能推理治理架构

Flutter 组件 google_generative_language_api 适配鸿蒙 HarmonyOS 实战:生成式 AI 集成,构建大语言模型调度与全场景智能推理治理架构

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 组件 google_generative_language_api 适配鸿蒙 HarmonyOS 实战:生成式 AI 集成,构建大语言模型调度与全场景智能推理治理架构 前言 在鸿蒙(OpenHarmony)生态迈向全场景 AI 赋能、涉及高效的语义理解、自动化内容生成及严苛的端云协同智能隐私保护背景下,如何实现一套既能深度对接 Google 生成式语言模型(如 Gemini、PaLM)、又能保障异步请求高响应性且具备多模态输入处理能力的“AI 调度中枢”,已成为决定应用智能化水平与用户体验代差的关键。在鸿蒙设备这类强调分布式协同与端侧算力按需分配的环境下,如果应用依然采用低效的 REST 手写拼接,由于由于 payload 结构复杂性,极易由于由于“协议解析异常”导致鸿蒙应用在大模型推理环节发生由于由于由于由于通讯阻塞。 我们需要一种能够统一模型调用语义、支持流式(Streaming)响应且符合鸿蒙异步异步并发范式的

【深度解析 Anthropic Claude-Code 2.1.88 源码结构:从 Source Map 揭秘 AI 编程助手内部实现】

前言 近日,一个开源项目在 GitHub 上引起了广泛关注。这个名为 claude-code-sourcemap 的项目通过技术手段还原了 Anthropic 官方 Claude-Code 工具 2.1.88 版本的源代码。作为技术研究者和 AI 编程工具的爱好者,我深入分析了这个项目,为大家带来详细的源码结构解析。 项目概况 项目名称:claude-code-sourcemap GitHub 地址:https://github.com/xy200303/claude-code-sourcemap 版本:2.1.88(基于 @anthropic-ai/claude-code npm 包) 文件数量:4756 个文件(包含 1884 个 .ts/.tsx 源文件) 还原方式:

别再把 AI 当聊天机器人了!手把手教你搭建“人机共生”的赛博办公室

别再把 AI 当聊天机器人了!手把手教你搭建“人机共生”的赛博办公室

目录 先说结论:10 分钟搭起“人机共管”的赛博办公室 一、整体部署与使用流程(先看全貌) 二、项目背景:DeskClaw 在解决什么问题? 三、环境准备:Docker Desktop 是唯一核心 步骤 1:安装 Docker Desktop 步骤 2:准备代码目录与项目文件 四、一键启动:Docker Compose 部署 步骤 1:启动服务 步骤 2:获取初始账号密码 步骤 3:浏览器访问平台 五、安装完成后的基本操作 六、实战示例:让 AI 同事每天帮你整理行业情报 场景设定 步骤 1: