IMU姿态解算,从IMU数据中计算旋转、速度、位置,IMU测量的原理

更新(2024.2.27):提供一个代码实现,供参考学习:>>戳我跳转<<

0. 预备

a. IMU测量值解释
IMU在测量时,得到的角速度或者加速度均是相对于地心惯性系结果,并且将该结果表示到Body坐标系下,就形成了最终的IMU输出。

记作: ω i b b \omega_{ib}^b ωibb,表示body系相对于惯性系的角速度测量在body下的表示。

a i b b a_{ib}^b aibb,表示body系相对于惯性系的加速度测量在body系下的表示。

b. IMU加速度计误差建模
a ~ i b b = a i b b + g b + b a + n a \tilde a_{ib}^b = a_{ib}^b + g^b + b^a + n^a a~ibb=aibb+gb+ba+na
其中: a ~ i b b \tilde a_{ib}^b a~ibb 表示的是IMU最终输出的测量值;

a i b b a_{ib}^b aibb 表示的是当前的加速度真值,去除了各种误差;

g b g^b gb重力加速度在body系下的表示;

b a b^a ba 加速度计的bias,它是一个恒定偏差,并且随时间缓慢变化;

n a n^a na加速度计的随机测量噪声,建模为0均值高斯噪声。

对于上式的加速度计建模,有时候也会建模到导航系下,多数情况下重力加速度是在导航系下有比较简洁的表示 [ 0 , 0 , g ] T [0, 0, g]^T [0,0,g]T,因此如果将加速度计的测量转到导航下就可以直接在导航系中简单的处理重力加速度。因此在实际使用时,有时也可以建模为如下形式:
a ~ i b b = R b w ( a i b w + g w ) + b a + n a \tilde a_{ib}^b = R_{bw} (a_{ib}^w + g^w) + b^a + n^a a~ibb=Rbw(aibw+gw)+ba+na
c. IMU陀螺仪误差建模
ω ~ i b b = ω i b b + b g + n g \tilde \omega_{ib}^b = \omega_{ib}^b + b^g + n^g ω~ibb=ωibb+bg+ng
其中: ω ~ i b b \tilde \omega_{ib}^b ω~ibb表示的是IMU最终输出的角速度测量值;

ω i b b \omega_{ib}^b ωibb表示的是当前的角速度真值,去除了各种误差;

b g b^g bg 陀螺仪的bias,并且随时间缓慢变化,短时间内可以认为是一个恒定偏差;

n g n^g ng 陀螺仪的随机测量噪声,建模为0均值高斯噪声。

d. 考虑地球自转时的各个角速度关系
ω i n n = ω i e n + ω e n n \omega_{in}^n = \omega_{ie}^n + \omega_{en}^n ωinn=ωien+ωenn
其中: ω i n n \omega_{in}^n ωinn 表示导航系相对于惯性系的角速度在导航下的表示

ω i e n \omega_{ie}^n ωien表示地球相对于惯性系的自转角速度在导航下的表示

ω e n n \omega_{en}^n ωenn 表示导航系相对于地球的角速度在导航下的表示

IMU测量的角速度是相对于惯性系i,而多数情况下进行导航姿态计算时是在导航系下的,也就是说如果直接使用IMU的角速度进行解算,就会把地球自转以及导航系变化带来的角速度都会引入到系统中,在高精度的IMU中这会带来误差,对于MEMS的IMU而言,可以忽略这一点。

对于 ω i e n \omega_{ie}^n ωien的计算如下
ω i e n = [ 0 ω i e cos ⁡ L   ω i e sin ⁡ L ] \omega_{ie}^n = [0 \quad \omega_{ie} \cos L \ \quad \omega_{ie} \sin L] ωien=[0ωiecosL ωiesinL]
其中: ω i e \omega_{ie} ωie是地球的自转角速度,取 7.2921151467 × 1 0 − 5 r a d / s 7.2921151467 \times 10^{-5} rad/s 7.2921151467×105rad/s。参考《捷联惯导算法与组合导航原理》 P54

L L L是当前位置对应的地理纬度

对于 ω e n n \omega_{en}^n ωenn的计算如下
ω e n n = [ − v N R M + h v E R N + h v E R N + h tan ⁡ L ] \omega_{en}^n = [-\frac{v_N}{R_M + h} \quad \frac{v_E}{R_N + h} \quad \frac{v_E}{R_N+h}\tan L] ωenn=[RM+hvNRN+hvERN+hvEtanL]
其中: v E v_E vE, v N v_N vN是当前时刻载体在东北天坐标系下的速度分量

R M , R N R_M,R_N RM,RN分别为子午圈主曲率半径和卯酉圈曲率半径 R N R_N RN,计算方法如下:参考《捷联惯导算法与组合导航原理》 P46和P48
R N = R e 1 − e 2 sin ⁡ 2 L R_N = \frac{R_e}{\sqrt{1 - e^2 \sin^2 L}} RN=1e2sin2L Re

R M = R e ( 1 − e 2 ) ( 1 − e 2 sin ⁡ 2 L ) 3 2 R_M = \frac{R_e(1-e^2)}{(1-e^2 \sin^2 L)^{\frac{3}{2}}} RM=(1e2sin2L)23Re(1e2)

L , h L,h L,h分别是当前位置对应的地理纬度和高度

注意:上面介绍了IMU的误差建模,但是后续的推导均未考虑误差

1. 姿态更新

参考《捷联惯导算法与组合导航原理》 P79

a.惯性系下的IMU姿态解算

C ˙ b i = C b i ∗ ( ω i b b × ) \dot C_b^i = C_b^i * (\omega_{ib}^b \times) C˙bi=Cbi(ωibb×)
上式是IMU的姿态的微分方程。如果要在惯性系下对上式进行解微分方程,即可得到IMU姿态计算结果。结果如下:

C b ( m ) i = C b ( m − 1 ) i C b ( m ) b ( m − 1 ) C_{b(m)}^{i} = C_{b(m-1)}^i C_{b(m)}^{b(m-1)} Cb(m)i=Cb(m1)iCb(m)b(m1)
其中: C b ( m ) i C_{b(m)}^{i} Cb(m)i 表示m时刻,b系到i的变换矩阵

C b ( m ) b ( m − 1 ) C_{b(m)}^{b(m-1)} Cb(m)b(m1) 表示由b系在m时刻到m-1时刻的变化矩阵,它就是有IMU的角速度测量计算而来。

需要格外的关注的是 C b ( m ) b ( m − 1 ) C_{b(m)}^{b(m-1)} Cb(m)b(m1)的计算,它是IMU姿态计算的关键,如果m-1时刻到m是IMU符合定轴转动,则由下式严格成立:

C b ( m ) b ( m − 1 ) = e ∫ t m − 1 t m ω i b b ( t ) d t C_{b(m)}^{b(m-1)} = e^{\int_{t_{m-1}}^{t_m} \omega_{ib}^b(t) dt} Cb(m)b(m1)=etm1tmωibb(t)dt
注意:上式是在定轴假设下才严格成立的,所谓定轴转动就是角速度的方向不发生变化,但是很显然这个假设在实际情况下不严格成立,但是幸运的是目前的IMU都是高频率的IMU它一次测量内很接近定轴转动。对于 e ∫ t m − 1 t m ω i b b ( t ) d t e^{\int_{t_{m-1}}{t_m} \omega_{ib}^b(t) dt} etm1tmωibb(t)dt的计算有多种方式,有欧拉积分,中值积分,矢量二子样算法,单子样+前一周期等算法。参考《捷联惯导算法与组合导航原理》 P28~P30。尽管解算方法很多,但是对于消费级的MEMS IMU使用中值积分就足够使用。

b.导航系下的IMU姿态解算

上面的过程是围绕着参考系为i系进行的,多数情况下我们在使用时都是用的导航系,下面就以东北线(ENU)导航系对IMU姿态解算进行推导介绍。导航下旋转矩阵对时间的微分方程如下:

C ˙ b n = C b n ∗ ( ω n b b × ) \dot C_b^n = C_b^n * (\omega_{nb}^b \times) C˙bn=Cbn(ωnbb×)
当考虑上地球转速以及导航系变化的角速度时, ω n b b = ω i b b − ω i n b \omega_{nb}^b = \omega_{ib}^b - \omega_{in}^b ωnbb=ωibbωinb,上式可以进一步列写为:

C ˙ b n = C b n ∗ ( ω n b b × ) = C b n [ ( ω i b b − ω i n b ) × ] = C b n ( ω i b b × ) − ( ω i n n × ) C b n \dot C_b^n = C_b^n * (\omega_{nb}^b \times) = C_b^n[(\omega_{ib}^b - \omega_{in}^b)\times] = C_b^n(\omega_{ib}^b \times) - (\omega_{in}^n\times)C_b^n C˙bn=Cbn(ωnbb×)=Cbn[(ωibbωinb)×]=Cbn(ωibb×)(ωinn×)Cbn
对上式进行求微分方程,并进行离散化就可以得到导航系下的姿态更新方法。但是求解比较麻烦,可以采用矩阵链式法则进行求解:

C b ( m ) n ( m ) = C n ( m − 1 ) n ( m ) C i n ( m − 1 ) C b ( m − 1 ) i C b ( m ) b ( m − 1 ) = C n ( m − 1 ) n ( m ) C b ( m − 1 ) n ( m − 1 ) C b ( m ) b ( m − 1 ) C_{b(m)}^{n(m)} = C_{n(m-1)}^{n(m)}C_{i}^{n(m-1)}C_{b(m-1)}^i C_{b(m)}^{b(m-1)} = C_{n(m-1)}^{n(m)} C_{b(m-1)}^{n(m-1)} C_{b(m)}^{b(m-1)} Cb(m)n(m)=Cn(m1)n(m)Cin(m1)Cb(m1)iCb(m)b(m1)=Cn(m1)n(m)Cb(m1)n(m1)Cb(m)b(m1)
其中: C n ( m − 1 ) n ( m ) C_{n(m-1)}^{n(m)} Cn(m1)n(m) 表示m-1时刻到m时刻导航系发生的姿态变化,也相当于在m时刻对应的导航系下表示m-1时刻导航系的位姿,计算方法是使用 ω i n n = ω i e n + ω e n n \omega_{in}^n = \omega_{ie}^n + \omega_{en}^n ωinn=ωien+ωenn.

C b ( m − 1 ) n ( m − 1 ) C_{b(m-1)}^{n(m-1)} Cb(m1)n(m1) 表示m-1时刻IMU相对于导航系的姿态

C b ( m ) b ( m − 1 ) C_{b(m)}^{b(m-1)} Cb(m)b(m1) 表示m时刻到m-1时刻IMU发生的位姿变化,它来自于IMU角速度测量的积分。

c. 导航系简化版姿态计算

C b ( m ) n ( m ) = C n ( m − 1 ) n ( m ) C b ( m − 1 ) n ( m − 1 ) C b ( m ) b ( m − 1 ) C_{b(m)}^{n(m)} = C_{n(m-1)}^{n(m)} C_{b(m-1)}^{n(m-1)} C_{b(m)}^{b(m-1)} Cb(m)n(m)=Cn(m1)n(m)Cb(m1)n(m1)Cb(m)b(m1)
在MEMS IMU中上式可以忽略导航系的运动 C n ( m − 1 ) n ( m ) C_{n(m-1)}^{n(m)} Cn(m1)n(m),因为消费级普通IMU无法敏感到这一项,所以可以忽略。于是可以得到简化版的姿态解算:

C b ( m ) n ( m ) = C b ( m − 1 ) n ( m − 1 ) C b ( m ) b ( m − 1 ) C_{b(m)}^{n(m)} = C_{b(m-1)}^{n(m-1)} C_{b(m)}^{b(m-1)} Cb(m)n(m)=Cb(m1)n(m1)Cb(m)b(m1)
其中, C b ( m ) b ( m − 1 ) = e ∫ t m − 1 t m ω i b b ( t ) d t C_{b(m)}^{b(m-1)} = e^{\int_{t_{m-1}}^{t_m} \omega_{ib}^b(t) dt} Cb(m)b(m1)=etm1tmωibb(t)dt,对于它的求解与前述思路一样,可以采用欧拉积分,中值积分等求解 ∫ t m − 1 t m ω i b b ( t ) d t {\int_{t_{m-1}}^{t_m} \omega_{ib}^b(t) dt} tm1tmωibb(t)dt,然后使用罗德里格斯公式完成指数映射。

2. 速度更新

参考《捷联惯导算法与组合导航原理》 P81

a. 精确速度微分建模

v ˙ e n n = C b n f s f b − ( 2 w i e n + w e n n ) × v e n n + g n \dot v_{en}^n = C_b^n f_{sf}^b - (2w_{ie}^n+w_{en}^n) \times v_{en}^n + g^n v˙enn=Cbnfsfb(2wien+wenn)×venn+gn
其中: f s f b f_{sf}^b fsfb 为加速度计测量的比力

− ( 2 w i e n + w e n n ) × v e n n - (2w_{ie}^n+w_{en}^n) \times v_{en}^n (2wien+wenn)×venn 为有害加速度

g n g^n gn 为重力加速度

以上便是对导航系下载体速度的精确微分建模,对其积分即可得到速度的解算,具体过程比较复杂,参考《捷联惯导算法与组合导航原理》 P82。

在IMU加速度误差的建模中使用的是加速度一词,这里使用的是比力,实际上可以认为它们是同样的意义。

b. 简化版速度微分建模

在MEMS IMU中,有害加速度这一项可以忽略,并忽略一些类似划桨误差等,可以得到简化版的速度更新方程:

v ˙ e n n = C b n f s f b + g n \dot v_{en}^n = C_b^n f_{sf}^b + g^n v˙enn=Cbnfsfb+gn
上式求积分之后改写成递推的形式为:

v m n ( m ) = v m − 1 n ( m − 1 ) + Δ v s f ( m ) n v_{m}^{n(m)} = v_{m-1}^{n(m-1)} + \Delta v_{sf(m)}^n vmn(m)=vm1n(m1)+Δvsf(m)n
其中: v m − 1 n ( m − 1 ) v_{m-1}^{n(m-1)} vm1n(m1)为m-1时刻载体速度在导航下的表示。

Δ v s f ( m ) n = ∫ m − 1 m ( C b n f s f b + g n ) d t \Delta v_{sf(m)}^n = \int_{m-1}^m (C_b^n f_{sf}^b + g^n)dt Δvsf(m)n=m1m(Cbnfsfb+gn)dt
上式的求解可以采用欧拉积分,中值积分等完成。

3. 位置更新

对于位置的更新,在参考《捷联惯导算法与组合导航原理》 P87 中采用的是经纬高来表示位置。而实际使用中多以导航系原点来表示位置,因此此处不总结介绍经纬高的方式。将直接在导航系下进行解算。

p ˙ n ( m ) = v n ( m ) \dot p^{n(m)} = v^{n(m)} p˙n(m)=vn(m)
对上式进行欧拉积分,即可得到递推方程:

p n ( m ) = p n ( m − 1 ) + ∫ m − 1 m v n ( t ) d t p^{n(m)} = p^{n(m-1)} + \int_{m-1}^m v^{n}(t)dt pn(m)=pn(m1)+m1mvn(t)dt
同样的方式,可以采用欧拉积分,中值积分等对 ∫ m − 1 m v n ( t ) d t \int_{m-1}^m v^{n}(t)dt m1mvn(t)dt进行求解。

例如,使用中值积分,并假设m-1到m时刻速度为线性变化,则有如下式:

p n ( m ) = p n ( m − 1 ) + v n ( m − 1 ) ∗ Δ t + 1 2 ( f s f n ( m − 1 ) + g n ) Δ t 2 p^{n(m)} = p^{n(m-1)} + v^{n(m-1)}*\Delta t + \frac{1}{2} (f_{sf}^{n(m-1)} +g^n) \Delta t^2 pn(m)=pn(m1)+vn(m1)Δt+21(fsfn(m1)+gn)Δt2
注意:IMU的姿态解算过程比较复杂,在建模时可以做的非常精确,这会导致模型非常复杂,实际当中使用时,普通的MEMS IMU无需做复杂的建模,不需要考虑地球转速,导航系的运动,速度划桨误差等等。当我们有了IMU初始时刻的姿态、速度、位置之后,就可以使用上述的递推方式计算出后续的姿态变化。但是对于低精度的IMU这个递推的过程维持不了多久就会发散。

  • 20
    点赞
  • 121
    收藏
    觉得还不错? 一键收藏
  • 7
    评论
由于IMU姿态解算涉及到许多数学知识和算法,因此需要一定的数学和编程基础才能进行。以下是一个简单的基于卡尔曼滤波的IMU姿态解算MATLAB代码示例: ```matlab % IMU姿态解算MATLAB代码示例 % 初始化状态向量,包括姿态、角速度和角度偏差 x = [0 0 0 0 0 0 0 0 0]'; % 初始化状态协方差矩阵 P = eye(9); % 初始化IMU测量误差协方差矩阵 R = diag([0.1 0.1 0.1 0.01 0.01 0.01]); % 初始化IMU测量噪声 Q = diag([0.001 0.001 0.001 0.0001 0.0001 0.0001]); % 初始化时间步长 dt = 0.01; % 加载IMU数据 load imu_data.mat; % 计算IMU数据长度 n = length(imu_data); % 初始化姿态角度 pitch = zeros(n,1); roll = zeros(n,1); yaw = zeros(n,1); % 开始姿态解算 for i=2:n % 计算角度变化 wx = imu_data(i,1); wy = imu_data(i,2); wz = imu_data(i,3); ax = imu_data(i,4); ay = imu_data(i,5); az = imu_data(i,6); [pitch(i),roll(i),yaw(i)] = imu_filter(wx, wy, wz, ax, ay, az, dt, x); end % 绘制姿态角度图像 plot(pitch); hold on; plot(roll); plot(yaw); legend('pitch','roll','yaw'); xlabel('Sample'); ylabel('Angle (deg)'); % 定义姿态解算函数 function [pitch,roll,yaw] = imu_filter(wx, wy, wz, ax, ay, az, dt, x) % 计算旋转矩阵和旋转角度 R = rotx(x(1))*roty(x(2))*rotz(x(3)); [yaw,pitch,roll] = dcm2angle(R); pitch = pitch*180/pi; roll = roll*180/pi; yaw = yaw*180/pi; % 构建状态转移矩阵 A = [eye(3) -R*dt zeros(3,3); zeros(3,3) eye(3) -dt*eye(3); zeros(3,3) zeros(3,3) eye(3)]; % 构建控制矩阵 B = [zeros(3,3);eye(3);zeros(3,3)]; % 计算预测状态和协方差矩阵 x = A*x + B*[wx;wy;wz;ax;ay;az]; P = A*P*A' + Q; % 构建测量矩阵 H = [zeros(3,6) eye(3)]; % 计算卡尔曼增益 K = P*H'/(H*P*H' + R); % 计算测量残差 z = [ax;ay;az]; z_hat = H*x; y = z - z_hat; % 更新状态和协方差矩阵 x = x + K*y; P = (eye(9) - K*H)*P; % 返回姿态角度 pitch = x(1); roll = x(2); yaw = x(3); end % 定义旋转矩阵函数 function R = rotx(theta) c = cosd(theta); s = sind(theta); R = [1 0 0; 0 c -s; 0 s c]; end function R = roty(theta) c = cosd(theta); s = sind(theta); R = [c 0 s; 0 1 0; -s 0 c]; end function R = rotz(theta) c = cosd(theta); s = sind(theta); R = [c -s 0; s c 0; 0 0 1]; end ``` 该代码使用了卡尔曼滤波算法进行IMU姿态解算,包括状态预测、测量更新和状态更新等步骤。通过加载IMU数据计算角度变化并传入姿态解算函数进行处理,最终得到姿态角度数据并绘制出图像。
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值