ESP32-S3-DevKitC-1开发记录帖——与MPU6050进行姿态检测

目录

MPU6050传感器——姿态检测

1.姿态检测

1.1 基本认识

1)坐标系

2)姿态角的关系

3)陀螺仪检测的缺陷

4)利用加速度计检测角度

5)利用磁场检测角度

1.2 姿态融合与四元数

1.3传感器工作原理

1.4 MPU6050模块简介

1.5 实践是检验真理的唯一标准

1.5.1 连接引脚确认

1.5.2 代码编写调试并运行


MPU6050传感器——姿态检测

1.姿态检测

1.1 基本认识

1)坐标系

分为载体坐标系、地理坐标系和地球坐标系

载体坐标系:以运载体的质心为原点,一般根据运载体自身结构方向构成坐标系

地理坐标系:原点在地球表面,Z轴沿当地地理垂涎的方向,XY轴沿当地经纬线的切线方向。

地球坐标系:以地球球心为原点,Z轴沿地球自转轴方向,XY轴在赤道平面内的坐标系

2)姿态角的关系

偏航角、俯仰角和横滚角

3)陀螺仪检测的缺陷

陀螺仪测量角度时需要使用积分,会存在一定的积分误差。而根据积分的性质可知,积分时间越小,误差就越小。所以可以用提高陀螺仪传感器的采样频率来减少积分误差。

4)利用加速度计检测角度

一般使用加速度传感器来检测倾角,通过检测器件在各个方向的形变情况而采样得到受力数据,根据F=ma转换,传感器直接输出加速度数据。

而在加速度传感器静止的时候,测出来的是加速度g,表示传感器在该方向作加速度为g的运动

那么是如何检测陀螺仪测出来的角度是否正确呢?

利用各方向的测量结果,根据力的分解原理,可以求出各个坐标轴与重力之间的夹角。因为重力方向是与地理坐标系的“天地”轴固连的,所以通过测量载体坐标系各轴与重力方向的角度即可修正角度

5)利用磁场检测角度

为了弥补加速度传感器无法检测偏航角的问题,通过检测地球磁场可实现指南针的功能。由于地磁场与地理坐标系的南北轴固联,利用磁场检测传感器的指南针功能

1.2 姿态融合与四元数

我们同时使用这两种传感器,并设计一个滤波算法,当物体处于静止状态时,增大加速度数据的权重,当物体处于运动状时,增大陀螺仪数据的权重,从而获得更准确的姿态数据。同理,检测偏航角,当载体在静止状态时,可增大磁场检测器数据的权重,当载体在运动状态时,增大陀螺仪和GPS检测数据的权重。这些采用多种传感器数据来检测姿态的处理算法被称为姿态融合。

1.3传感器工作原理

敏感元件:直接感受被测物理量,并输出与该物理量有确定关系的信号

转换原件:将物理信号转换为电信号量

变换电路:对转换元件输出的电信号进行放大调制,最后输出容易检测的电信号量

1.4 MPU6050模块简介

功能:检测三轴加速度、三轴陀螺仪的运动数据以及温度数据

引脚说明:

如何使用MPU6050传输和计算数据?

可以使用STM32控制器把这些数据读取出来然后进行姿态融合解算,以求出传感器当前的姿态。

也可以使用传感器内部的DMP单元进行解算,可以直接对采样得到的加速度及角速度进行姿态解算。

1.5 实践是检验真理的唯一标准

1.5.1 连接引脚确认

我首先查了MPU每个引脚具体该怎么连接ESP32,在这个博客中看到了这样一句话:

“使用的是IO19与IO20两个引脚,使用杜邦线将开发板的这两个引脚分别接到mpu6050模块的SCL与SDA引脚上,将ADO引脚接到GND。

ADO引脚为从机地址设置引脚,接地或悬空时, 地址为: 0x68;接VCC时,地址为:0x69”

而该博主用的是ESP32-S3-LCD-EV-Board-MB开发板,所以我搜了一下他的引脚:

根据ESP32-DevKitc-1的引脚说明书,找到符合I2C协议的引脚。

得出结论:

名称为21的引脚连接MPU6050的SCL,名称为20连接MPU6050的SDA

1.5.2 代码编写调试并运行

调试代码教程:

烧录的时候出现了以下错误:

结果是自己电脑端口出现了问题...在别人电脑上却能跑通。

点击调试和运行运行代码:

#include <Wire.h>
#include <MPU6050.h>
#include <WiFi.h>
#include <HTTPClient.h>
#include <ArduinoJson.h>

#define SDA_PIN 21
#define SCL_PIN 22

MPU6050 mpu6050(0x68); // 使用默认地址 0x68
unsigned long last_sample_time = 0;
int peak_count = 0;
bool condition_met = false;

int16_t ax,ay,az,gx,gy,gz;

void setup() {
  Serial.begin(115200);
  Wire.begin(SDA_PIN, SCL_PIN);

  // 校准陀螺仪偏移
  mpu6050.setXGyroOffset(220);
  mpu6050.setYGyroOffset(76);
  mpu6050.setZGyroOffset(-85);

  last_sample_time = millis();
}

void loop() {
  mpu6050.getMotion6(&ax, &ay, &az, &gx, &gy, &gz);
  
  // 将原始数据转换为实际值
  float ax_g = ax * 2.0 / 32768.0;
  float ay_g = ay * 2.0 / 32768.0;
  float az_g = az * 2.0 / 32768.0;
  float gx_rad = gx * 250.0 / 32768.0 * 0.0174533;
  float gy_rad = gy * 250.0 / 32768.0 * 0.0174533;
  float gz_rad = gz * 250.0 / 32768.0 * 0.0174533;

  // 检查加速度是否超过阈值
  if (abs(ax_g) > 1.0 || abs(ay_g) > 1.0 || abs(az_g) > 1.0) {
    peak_count++;
  }

  // 检查是否达到采样周期
  unsigned long current_time = millis();
  if (current_time - last_sample_time >= 1000) { // 1秒采样一次
    // 计算峰值频率
    float peak_frequency = (peak_count * 60.0) / (current_time - last_sample_time);
    peak_count = 0;

    // 检查峰值频率是否在指定范围内
    if (peak_frequency >= 100 && peak_frequency <= 120) {
      condition_met = true;
    }

    last_sample_time = current_time;

    // 如果条件满足,发送完成信息到后端
    if (condition_met) {
      sendCompletionToBackend(ax_g, ay_g, az_g, gx_rad, gy_rad, gz_rad);
      condition_met = false;
    }
  }

  // 打印数据
  Serial.print("ax: ");
  Serial.print(ax_g);
  Serial.print(" g\t");
  Serial.print("ay: ");
  Serial.print(ay_g);
  Serial.print(" g\t");
  Serial.print("az: ");
  Serial.print(az_g);
  Serial.print(" g\t");
  Serial.print("gx: ");
  Serial.print(gx_rad);
  Serial.print(" rad/s\t");
  Serial.print("gy: ");
  Serial.print(gy_rad);
  Serial.print(" rad/s\t");
  Serial.print("gz: ");
  Serial.println(gz_rad);
  Serial.print("\n");

  delay(10); // 避免循环过快
}

void sendCompletionToBackend(float ax, float ay, float az, float gx, float gy, float gz) {
  Serial.println("Sending data to backend...");

  // 连接WiFi
  WiFi.begin(ssid, password);
  while (WiFi.status() != WL_CONNECTED) {
    delay(500);
    Serial.print(".");
  }

  // 创建HTTP客户端请求
  HTTPClient http;
  http.begin(serverUrl);
  http.addHeader("Content-Type", "application/json");

  // 构建JSON数据
 StaticJsonDocument<200> jsonDoc; // 根据需要调整JSON文档的大小
  jsonDoc["ax"] = ax;
  jsonDoc["ay"] = ay;
  jsonDoc["az"] = az;
  jsonDoc["gx"] = gx;
  jsonDoc["gy"] = gy;
  jsonDoc["gz"] = gz;

  String jsonData;
  serializeJson(jsonDoc, jsonData); // 序列化JSON

  // 发送POST请求
  int httpResponseCode = http.POST(jsonData);
  if (httpResponseCode > 0) {
    String response = http.getString();
    Serial.println("Response from server: ");
    Serial.println(response);
  } else {
    Serial.println("Error on sending POST request");
  }
  http.end();
}

### ESP32MPU6050使用DMP进行姿态解算 #### 初始化设置 对于ESP32连接到MPU6050并通过DMP实现姿态解算,初始化过程至关重要。这包括配置I2C接口以及确保能够成功激活和启动DMP模块。 ```cpp #include <Wire.h> #include "I2Cdev.h" #include "MPU6050_6Axis_MotionApps20.h" // 设备地址定义 #define MPU6050_ADDRESS 0x68 MPU6050 mpu; void setup() { Wire.begin(); Serial.begin(115200); // 加载DMP固件并验证其存在性 devStatus = mpu.dmpInitialize(); // 验证连接状态 if (devStatus == 0) { mpu.CalibrateGyro(5); mpu.setDMPEnabled(true); mpu.InterruptMode(MPU6050_INTERRUPT_DMP_READY); // 开启DMP中断使能位 mpu.SetIntEnable(MPU6050_INTERRUPT_DMP_INT); } } ``` 此部分代码负责建立硬件间的通信链路,并准备就绪以接收来自传感器的数据流[^1]。 #### 数据处理循环 一旦设备被正确初始化,则可以在主程序循环内定期获取由DMP计算得出的姿态信息: ```cpp uint8_t fifoBuffer[64]; Quaternion q; VectorFloat gravity; float ypr[3]; void loop() { while (!mpu.dmpGetCurrentFIFOPacket(fifoBuffer)) {} mpu.dmpGetQuaternion(&q, fifoBuffer); mpu.dmpGetGravity(&gravity, &q); mpu.dmpGetYawPitchRoll(ypr, &q, &gravity); Serial.print("yaw: "); Serial.print(ypr[0] * 180/M_PI); Serial.print(", pitch: "); Serial.print(ypr[1] * 180/M_PI); Serial.print(", roll: "); Serial.println(ypr[2] * 180/M_PI); delay(100); } ``` 上述代码片段展示了如何解析从DMP得到的信息包,并将其转换成易于理解的角度表示形式——即偏航角(Yaw)、俯仰角(Pitch)和横滚角(Roll),这些值通常用于描述物体的空间方位变化情况[^3]。 #### 解决可能遇到的问题 值得注意的是,在实际操作过程中可能会碰到一些挑战,比如初始化阶段容易出现卡死现象。针对这种情况可以借鉴已有的解决方案,例如参考特定平台上的成熟案例研究或官方文档中的建议做法来优化现有流程[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值