IMU惯性里程计解算(附代码实现)

一、系统概述

IMU是机器人常用的传感器之一,IMU对机器人的定位功能实现非常重要,其优点在于是内源传感器对外部环境变化不明显,输出频率高,缺点在于存在累积误差。本文主要记录一下在机器人定位中对IMU的使用和对惯性导航里程计的理解和实现。

本文代码主要依赖于ROS相关库实现,源代码见:GitHub - Abin1258/imu_to_odom: imu odometry

1.系统输入:IMU传感器测量数据:线性加速度、角速度

在ROS消息中的格式为:

ps:1.要注意观察不同IMU传感器的单位不同,有的传感器加速度单位是重力加速度的倍数,有的传感器是米每秒的平方,本文所用传感器的单位是

2.系统输出:Odometry消息:位置、姿态、线速度、角速度和每个量的协方差矩阵

在ROS消息中的格式为:

​3.解算模型:

3.1初始化

3.1.1 0时刻位姿初始化

一般设置0时刻为里程计起点,即位姿速度全为0,代码上如下:

odom.header.frame_id = "odom"; odom.child_frame_id = "base_link"; Eigen::Vector3d zero(0, 0, 0); point.pos = zero; point.orien = Eigen::Matrix3d::Identity(); point.v = zero; point.w = zero; firstT = true;

3.1.2 重力初始化

只有IMU一个传感器,所以直接用了第一帧数据(假设当前载体处于静止状态)的加速度作为重力加速度项,代码如下:

gravity[0] = msg.x; gravity[1] = msg.y; gravity[2] = msg.z;

3.2 求解位姿

初始化完成后,先求解位姿,因为求解位置的时候需要使用位姿结果将IMU坐标系下的加速度转化到全局坐标系下的加速度,求解位姿的方法有很多,在下面的章节陆续补充,本文代码实现的是用旋转矩阵表示的方法求解的位姿,代码如下:

point.w << msg.x, msg.y, msg.z; //基于旋转矩阵表示方法 Eigen::Matrix3d B; B << 0, -msg.z * deltaT, msg.y * deltaT, msg.z * deltaT, 0, -msg.x * deltaT, -msg.y * deltaT, msg.x * deltaT, 0; //欧拉法 double sigma = std::sqrt(std::pow(msg.x, 2) + std::pow(msg.y, 2) + std::pow(msg.z, 2)) * deltaT; //罗德里格斯公式 point.orien = point.orien * (Eigen::Matrix3d::Identity() + (std::sin(sigma) / sigma) * B - ((1 - std::cos(sigma)) / std::pow(sigma, 2)) * B * B);

对应公式如下:

 

3.3 求解线速度和位置

求解完位姿后求解位置就较为简单,两个积分公式即可

  Eigen::Vector3d acc_l(msg.x, msg.y, msg.z);//imu坐标系下的加速度
  Eigen::Vector3d acc_g = point.orien * acc_l;//转化到里程计坐标系下的加速度
  point.v = point.v + deltaT * (acc_g - gravity);//积分得到速度
  point.pos = point.pos + deltaT * point.v;//积分得到位置

4.实现效果

求解完成后只需要发布里程计消息,即可在RVIZ中观测,实际效果如果没有其他传感器观测矫正误差,10秒中左右累积误差已经达到米级

二、IMU里程计原理和公式推倒

未完待续...

知乎链接:知乎 - 有问题,就会有答案

  • 4
    点赞
  • 56
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
下面是一个简单的 MATLAB 代码示例,用于将 IMU 数据与里程计数据融合。假设你已经有了 IMU 数据和里程计数据,并且它们都以某种方式进行了校准和预处理: ```matlab % IMU 数据 imu_data = load('imu_data.mat'); % 里程计数据 odo_data = load('odo_data.mat'); % 融合参数 dt = 0.01; % 时间间隔 Q = diag([0.01 0.01 0.01 0.1 0.1 0.1]); % IMU 噪声协方差矩阵 R = diag([0.1 0.1 0.1]); % 里程计噪声协方差矩阵 % 初始化状态向量和协方差矩阵 x = [0; 0; 0; 0; 0; 0]; P = eye(6); % 预测步骤(根据 IMU 数据更新状态) for i = 1:length(imu_data) [x, F] = predict_state(x, imu_data(i,:), dt); [P, Q] = predict_covariance(P, Q, F, dt); end % 更新步骤(根据里程计数据更新状态) for i = 1:length(odo_data) [x, H] = update_state(x, odo_data(i,:)); [P, R] = update_covariance(P, R, H); end % 预测状态函数(根据 IMU 数据更新状态) function [x, F] = predict_state(x, imu, dt) % imu: 加速度计和角速度计数据 % x: 状态向量 % dt: 时间间隔 % 根据加速度计和角速度计计算状态变化量 ax = imu(1); ay = imu(2); az = imu(3); wx = imu(4); wy = imu(5); wz = imu(6); dx = x(4); dy = x(5); dz = x(6); x_dot = [dx; dy; dz; -ax; -ay; -az] + [0; 0; 0; -wx; -wy; -wz]; % 计算状态转移矩阵 F F = [eye(3) dt*eye(3); zeros(3) eye(3)]; % 更新状态向量 x x = x + dt * x_dot; end % 预测协方差函数(根据 IMU 数据更新协方差) function [P, Q] = predict_covariance(P, Q, F, dt) % P: 协方差矩阵 % Q: IMU 噪声协方差矩阵 % F: 状态转移矩阵 % dt: 时间间隔 % 计算预测协方差矩阵 P = F * P * F' + Q; end % 更新状态函数(根据里程计数据更新状态) function [x, H] = update_state(x, odo) % odo: 里程计数据 % x: 状态向量 % 根据里程计计算状态变化量 dx = odo(1); dy = odo(2); dtheta = odo(3); x_dot = [dx; dy; dtheta; 0; 0; 0]; % 计算测量矩阵 H H = [eye(3) zeros(3)]; % 更新状态向量 x x = x + x_dot; end % 更新协方差函数(根据里程计数据更新协方差) function [P, R] = update_covariance(P, R, H) % P: 协方差矩阵 % R: 里程计噪声协方差矩阵 % H: 测量矩阵 % 计算卡尔曼增益 K K = P * H' / (H * P * H' + R); % 计算更新后的协方差矩阵 P P = (eye(6) - K * H) * P; % 更新里程计噪声协方差矩阵 R(可以根据实际情况调整) R = diag([0.1 0.1 0.1]); % 这里假设里程计的噪声是恒定的 end ``` 这段代码只是一个简单的示例,具体的实现可能需要根据你的具体应用情况进行调整和优化。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值