智能车制作

 找回密码
 注册

扫一扫,访问微社区

查看: 5887|回复: 19
打印 上一主题 下一主题

四轴飞行器&角度融合波形

[复制链接]

3

主题

17

帖子

0

精华

高级会员

Rank: 4

积分
912
威望
453
贡献
287
兑换币
286
注册时间
2012-12-9
在线时间
86 小时
毕业学校
XMUT
跳转到指定楼层
1#
发表于 2014-11-24 22:22:15 | 只看该作者 回帖奖励 |正序浏览 |阅读模式
本帖最后由 哈达恋 于 2014-11-24 22:23 编辑

  首先,先贴下我的姿态解算程序:
AHRSupdate四元数姿态解算:
  1. // Definitions

  2. #define Kp 6.0f                        // proportional gain governs rate of convergence to accelerometer/magnetometer
  3. #define Ki 0.03f                // integral gain governs rate of convergence of gyroscope biases
  4. #define halfT 0.0014                // half the sample period

  5. //---------------------------------------------------------------------------------------------------
  6. // Variable definitions

  7. float q0 = 1, q1 = 0, q2 = 0, q3 = 0;        // quaternion elements representing the estimated orientation
  8. float exInt = 0, eyInt = 0, ezInt = 0;        // scaled integral error

  9. //====================================================================================================
  10. // Function
  11. //====================================================================================================

  12. void AHRSupdate(float gx, float gy, float gz, float ax, float ay, float az, float mx, float my, float mz) {
  13.         float norm;
  14.         float hx, hy, hz, bx, bz;
  15.         float vx, vy, vz, wx, wy, wz;
  16.         float ex, ey, ez;

  17.         // auxiliary variables to reduce number of repeated operations
  18.         float q0q0 = q0*q0;
  19.         float q0q1 = q0*q1;
  20.         float q0q2 = q0*q2;
  21.         float q0q3 = q0*q3;
  22.         float q1q1 = q1*q1;
  23.         float q1q2 = q1*q2;
  24.         float q1q3 = q1*q3;
  25.         float q2q2 = q2*q2;   
  26.         float q2q3 = q2*q3;
  27.         float q3q3 = q3*q3;         
  28.         
  29.         // normalise the measurements
  30.         norm = sqrt(ax*ax + ay*ay + az*az);      
  31.         ax = ax/ norm;
  32.         ay = ay/ norm;
  33.         az = az/ norm;
  34.         norm = sqrt(mx*mx + my*my + mz*mz);         
  35.         mx = mx/ norm;
  36.         my = my/ norm;
  37.         mz = mz/ norm;         
  38.         
  39.         // compute reference direction of flux
  40.         hx = 2*mx*(0.5 - q2q2 - q3q3) + 2*my*(q1q2 - q0q3) + 2*mz*(q1q3 + q0q2);
  41.         hy = 2*mx*(q1q2 + q0q3) + 2*my*(0.5 - q1q1 - q3q3) + 2*mz*(q2q3 - q0q1);
  42.         hz = 2*mx*(q1q3 - q0q2) + 2*my*(q2q3 + q0q1) + 2*mz*(0.5 - q1q1 - q2q2);         
  43.         bx = sqrt((hx*hx) + (hy*hy));
  44.         bz = hz;        
  45.         
  46.         // estimated direction of gravity and flux (v and w)
  47.         vx = 2*(q1q3 - q0q2);
  48.         vy = 2*(q0q1 + q2q3);
  49.         vz = q0q0 - q1q1 - q2q2 + q3q3;
  50.         wx = 2*bx*(0.5 - q2q2 - q3q3) + 2*bz*(q1q3 - q0q2);
  51.         wy = 2*bx*(q1q2 - q0q3) + 2*bz*(q0q1 + q2q3);
  52.         wz = 2*bx*(q0q2 + q1q3) + 2*bz*(0.5 - q1q1 - q2q2);  
  53.         
  54.         // error is sum of cross product between reference direction of fields and direction measured by sensors
  55.         ex = (ay*vz - az*vy) + (my*wz - mz*wy);
  56.         ey = (az*vx - ax*vz) + (mz*wx - mx*wz);
  57.         ez = (ax*vy - ay*vx) + (mx*wy - my*wx);
  58.         
  59.         // integral error scaled integral gain
  60.         exInt = exInt + ex*Ki;
  61.         eyInt = eyInt + ey*Ki;
  62.         ezInt = ezInt + ez*Ki;
  63.         
  64.         // adjusted gyroscope measurements
  65.         gx = gx + Kp*ex + exInt;
  66.         gy = gy + Kp*ey + eyInt;
  67.         gz = gz + Kp*ez + ezInt;
  68.         
  69.         // integrate quaternion rate and normalise
  70.         q0 = q0 + (-q1*gx - q2*gy - q3*gz)*halfT;
  71.         q1 = q1 + (q0*gx + q2*gz - q3*gy)*halfT;
  72.         q2 = q2 + (q0*gy - q1*gz + q3*gx)*halfT;
  73.         q3 = q3 + (q0*gz + q1*gy - q2*gx)*halfT;  
  74.         
  75.         // normalise quaternion
  76.         norm = sqrt(q0*q0 + q1*q1 + q2*q2 + q3*q3);
  77.         q0 = q0 / norm;
  78.         q1 = q1 / norm;
  79.         q2 = q2 / norm;
  80.         q3 = q3 / norm;
  81. }
复制代码
卡尔曼滤波:
  1. //============================================================================
  2. //                 卡尔曼滤波参数
  3. float Q_angle=0.001,
  4.       Q_gyro=0.005,
  5.       R_angle=0.5,
  6.       dt=0.0055,//注意:dt的取值为kalman滤波器采样时间;
  7.       angle,
  8.       angle_dot;   

  9. float P[2][2] = {            { 1, 0 },
  10.                              { 0, 1 }
  11.                  };                  //alpha_k
  12.    
  13. float Pdot[4] ={0,0,0,0};        //Pdot = (u_k-bias)
  14. const char C_0 = 1;       //H = [1 0]矩阵,用于提取
  15. float   q_bias ,           //陀螺仪常值偏差bias
  16.         angle_err,        //观测值减去估计值
  17.         PCt_0, PCt_1,     //计算K的中间步骤,分子 PCT = PH_T
  18.         E,                //计算K的中间步骤,分母 E =(HPH_T + R)
  19.         K_0, K_1,         //K为卡尔曼增益,K = P(K),矩阵
  20.         t_0, t_1;         //更新alpha的中间步骤  




  21. //============================================================================
  22. //函数名称:Kalman_Filter
  23. //功能概要:卡尔曼滤波

  24. void Kalman_Filter(float angle_m,float gyro_m)          //gyro_m:gyro_measure
  25. {

  26.   


  27. //        卡尔曼滤波开始
  28.    
  29.     angle+=(gyro_m-q_bias) * dt;//先验估计


  30.     Pdot[0]=Q_angle - P[0][1] - P[1][0];// Pk-' 先验估计误差协方差的微分
  31.     Pdot[1]=- P[1][1];
  32.     Pdot[2]=- P[1][1];
  33.     Pdot[3]=Q_gyro;                      //Pdot = (u_k-bias)
  34.    
  35.     P[0][0] += Pdot[0] * dt;        // Pk- 先验估计误差协方差微分的积分 = 先验估计误差协方差
  36.     P[0][1] += Pdot[1] * dt;
  37.     P[1][0] += Pdot[2] * dt;
  38.     P[1][1] += Pdot[3] * dt;        //alpha_k = alpha_k-1 + [(u_k-bias)*dt]
  39.    
  40.    
  41.     angle_err = angle_m - angle;    //zk-先验估计 观测值减去估计值
  42.    
  43.    
  44.     PCt_0 = C_0 * P[0][0];
  45.     PCt_1 = C_0 * P[1][0];          //计算K的中间步骤,分子 PCT = PH^T
  46.    
  47.     E = R_angle + C_0 * PCt_0;      //计算K的中间步骤,分母 E =(HPH^T + R)
  48.    
  49.     K_0 = PCt_0 / E;        //K =   PH^T / (HPH^T + R)
  50.     K_1 = PCt_1 / E;
  51.    
  52.     t_0 = PCt_0;            //更新alpha的中间步骤
  53.     t_1 = C_0 * P[0][1];

  54.     P[0][0] -= K_0 * t_0;   //后验估计误差协方差
  55.     P[0][1] -= K_0 * t_1;
  56.     P[1][0] -= K_1 * t_0;   //alpha_k+1 = alpha_k - KH*alpha_k
  57.     P[1][1] -= K_1 * t_1;
  58.    
  59.    
  60.     angle+= K_0 * angle_err;//后验估计   更新估计值作为最终结果
  61.     q_bias+= K_1 * angle_err;//后验估计   更新bias的值
  62.     angle_dot=gyro_m-q_bias;//输出值(后验估计)的微分 = 角速度
  63.    

  64. //                 卡尔曼滤波结束



  65. }
复制代码
如上, 我使用了以上的两个函数进行实验进行姿态解算,卡尔曼滤波成功使用,已经在去年的华南赛区直立摄像头组跑完全程,直立上是妥妥的。由于快毕业了,比赛结束后没有休息,继续做下了四轴,两个月后也成功飞行~
  可是飞行的时候发现遥控起来特别吃力,很容易在空中荡秋千,超难维持稳定,一开始以为是PID调试问题,后面发现不是。这几天使用了MP6050的DMP功能,昨晚再次移植成功,四轴非常稳定飞行,达到了几乎快悬停的效果,尽管在狭窄的走廊也能操控。这时候我意识到了问题所在,是角度融合有问题!!
  以下是我用以上四元数的函数解算出来的角度波形



如上波形,第一图左边的波形相对还是比较完美的,原地直立是没有问题的,如果使用在直立车上是可以妥妥直立的。但是四轴是在空中飞行的,只要运动,就很容易叠加上运动的加速度,这时候加速度计和陀螺仪融合出来的角度就出错了。看第一图的右边,这是我在角度维持不变,进行平移加速突然停止的动作而呈现的波形,很明显看到运动的加速度成分的存在。
  我在做实验的时候虽然手拿会有波动,但不至于产生二三十度的波动,这能很明显的发现是由加速度产生的。那么问题就来了?!!!
  同样的姿态解算,这个四元数和卡尔曼已经是网上被移植很多的程序,不过大家的四轴好像都飞行得很好,难道就没有出现遥控难度难的荡秋千现象么?
  不知道在这论坛里是否有人尝试去解决这个问题或者发现这个问题的呢?你们是如何去解决的?


本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x
回复

使用道具 举报

2

主题

98

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2253
威望
1106
贡献
715
兑换币
740
注册时间
2016-6-9
在线时间
216 小时
毕业学校
XMUT
20#
发表于 2016-11-30 16:46:35 | 只看该作者
哈达恋 发表于 2016-11-29 21:25
我的理解和看法是这个滤波其实就是一个融合的过程,当然像加速度计的均值滤波不在其中,这个是真的在做低 ...

还有 学长 直立车直立环怎样算硬?拨一下它需要回来停住还是不需要回来停住?
回复 支持 反对

使用道具 举报

0

主题

770

帖子

0

精华

跨届大侠

Rank: 10Rank: 10Rank: 10

积分
9280
威望
5224
贡献
3388
兑换币
3559
注册时间
2016-9-1
在线时间
334 小时
毕业学校
西南某校
19#
发表于 2016-11-30 13:10:04 | 只看该作者
调试过程,谢谢楼主,支持分享
回复 支持 反对

使用道具 举报

2

主题

98

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2253
威望
1106
贡献
715
兑换币
740
注册时间
2016-6-9
在线时间
216 小时
毕业学校
XMUT
18#
发表于 2016-11-30 10:00:01 | 只看该作者
哈达恋 发表于 2016-11-29 21:25
我的理解和看法是这个滤波其实就是一个融合的过程,当然像加速度计的均值滤波不在其中,这个是真的在做低 ...

谢谢健力学长,我还要把直立环调硬一点。
回复 支持 反对

使用道具 举报

3

主题

17

帖子

0

精华

高级会员

Rank: 4

积分
912
威望
453
贡献
287
兑换币
286
注册时间
2012-12-9
在线时间
86 小时
毕业学校
XMUT
17#
 楼主| 发表于 2016-11-29 21:25:46 | 只看该作者
Traverlcq 发表于 2016-7-26 10:52
我现在不懂融合的算法怎么写,结构框图已经画出来了。融合在滤波之前还是之后?

我的理解和看法是这个滤波其实就是一个融合的过程,当然像加速度计的均值滤波不在其中,这个是真的在做低通滤波,卡尔曼是通过一种递归的方式将加速度计和陀螺仪融合起来,我是这么理解,我是还没有真正理解透就是
回复 支持 反对

使用道具 举报

3

主题

17

帖子

0

精华

高级会员

Rank: 4

积分
912
威望
453
贡献
287
兑换币
286
注册时间
2012-12-9
在线时间
86 小时
毕业学校
XMUT
16#
 楼主| 发表于 2016-11-29 21:23:45 | 只看该作者
Traverlcq 发表于 2016-7-26 10:41
XMUT 学长好,一眼看就猜到是创新的桌子

现在毕业了没有
回复 支持 反对

使用道具 举报

2

主题

98

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2253
威望
1106
贡献
715
兑换币
740
注册时间
2016-6-9
在线时间
216 小时
毕业学校
XMUT
15#
发表于 2016-7-26 10:52:06 | 只看该作者
我现在不懂融合的算法怎么写,结构框图已经画出来了。融合在滤波之前还是之后?
回复 支持 反对

使用道具 举报

2

主题

98

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
2253
威望
1106
贡献
715
兑换币
740
注册时间
2016-6-9
在线时间
216 小时
毕业学校
XMUT
14#
发表于 2016-7-26 10:41:35 | 只看该作者
XMUT 学长好,一眼看就猜到是创新的桌子
回复 支持 反对

使用道具 举报

3

主题

17

帖子

0

精华

高级会员

Rank: 4

积分
912
威望
453
贡献
287
兑换币
286
注册时间
2012-12-9
在线时间
86 小时
毕业学校
XMUT
13#
 楼主| 发表于 2014-12-23 20:28:49 | 只看该作者
叶鸣 发表于 2014-12-17 17:45
卡尔曼融合出来的角度是是那个轴的?

三个轴都进行了融合,用三个卡尔曼
回复 支持 反对

使用道具 举报

2

主题

64

帖子

0

精华

高级会员

Rank: 4

积分
855
威望
438
贡献
257
兑换币
290
注册时间
2014-2-17
在线时间
80 小时
毕业学校
西南交通大学峨眉校区
12#
发表于 2014-12-22 03:02:16 | 只看该作者
真虎!
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 注册

本版积分规则

关于我们|联系我们|小黑屋|智能车制作 ( 黑ICP备2022002344号

GMT+8, 2025-1-13 02:35 , Processed in 0.050295 second(s), 31 queries , Gzip On.

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表