极客工坊

 找回密码
 注册

QQ登录

只需一步,快速开始

查看: 11259|回复: 2

平衡车,求高手,我这段代码怎么实现不了前进后退?

[复制链接]
发表于 2013-3-22 15:20:48 | 显示全部楼层 |阅读模式
#include "Wire.h"
#include "I2Cdev.h"
#include "MPU6050.h"

MPU6050 accelgyro;
#define Gry_offset 674     
#define Gyr_Gain 65.5
#define pi 3.14159

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

/*********** PID控制器参数 *********/
float Output;   
float kp, ki, kd,kpp;
float angleA,omega;
float P[2][2] = {{ 1, 0 },{ 0, 1 }};
float Pdot[4] ={ 0,0,0,0};
static const double Q_angle=0.01, Q_gyro=0.03, R_angle=0.5,dtt=0.005,C_0 = 1;
float q_bias, angle_err, PCt_0, PCt_1, E, K_0, K_1, t_0, t_1;
float angle, angle_dot,aaxdot,aax;
float position_dot,position_dot_filter,positiono;
//--------------------------------------

//-----------control arguments-----------
int oommm;
int ooommm;

int TN1=2;
int TN2=3;
int TN3=4;
int TN4=7;
int ENA=5;
int ENB=6;

int A=A2;
int C=A3;
int B=A0;
int D=A1;


float LOutput;
float ROutput;
float LLOutput;
float RROutput;



//-------------------------------------
void setup() {
Wire.begin();
delay(1000);

accelgyro.initialize();
   delay(100);
pinMode(2,OUTPUT);         
    pinMode(3,OUTPUT);
      pinMode(4,OUTPUT);
        pinMode(7,OUTPUT);
              pinMode(5,OUTPUT);
        pinMode(6,OUTPUT);
        
delay(1);
//Serial.begin(4800);
}

void loop() {
//---------control-----------  
  int up=digitalRead(A);
  int down=digitalRead(C);
  int leftB=digitalRead(B);
  int rightD=digitalRead(D);
if(up==HIGH)
{
  former();

}
if(down==HIGH)
{
  back();
  
}
if(leftB==HIGH)
{
left();


}
if(rightD==HIGH)
{
  right();
  
}
//------control----------
    accelgyro.getMotion6(&ax, &ay, &az, &gx, &gy, &gz);
  //-----------------------------------------------------------------------------------------------------------------

  angleA= atan2(ay , az) * 180 / pi;  
  omega=(gx +  Gry_offset)/Gyr_Gain;
  if (abs(angleA)<30) {
  Kalman_Filter(angleA,omega);
  PIDD();
  PWMB();
   }
    else {
analogWrite(ENA, 0); //PWM调速a==0-255
analogWrite(ENB, 0);
  }
     delay(20);
     
    Serial.print(LOutput); Serial.print("\t");
    Serial.print(ROutput);Serial.print("\t");
    Serial.print(ax/10); Serial.print("\t");
    Serial.print(ay/10); Serial.print("\t");
    Serial.print(az/10); Serial.print("\t");
    Serial.print(gx/10); Serial.print("\t");
    Serial.print(gy/10); Serial.print("\t");
    Serial.print(gz/10);Serial.print("\t");

}
//=---------------------------------------------------------------
void left(){
  LLOutput=60;//BD
  RROutput=60;
}
void right(){
  LLOutput=-60;//BD
  RROutput=-60;
}
void back(){
  LLOutput=100;//C
  RROutput=-100;
}
void  former(){
   LLOutput=-100;//A
   RROutput=100;
}

//-----------------------------

void Kalman_Filter(double angle_m,double gyro_m)
{
angle+=(gyro_m-q_bias) * dtt;
Pdot[0]=Q_angle - P[0][1] - P[1][0];
Pdot[1]=- P[1][1];
Pdot[2]=- P[1][1];
Pdot[3]=Q_gyro;
P[0][0] += Pdot[0] * dtt;
P[0][1] += Pdot[1] * dtt;
P[1][0] += Pdot[2] * dtt;
P[1][1] += Pdot[3] * dtt;
angle_err = angle_m - angle;
PCt_0 = C_0 * P[0][0];
PCt_1 = C_0 * P[1][0];
E = R_angle + C_0 * PCt_0;
K_0 = PCt_0 / E;
K_1 = PCt_1 / E;
t_0 = PCt_0;
t_1 = C_0 * P[0][1];
P[0][0] -= K_0 * t_0;
P[0][1] -= K_0 * t_1;
P[1][0] -= K_1 * t_0;
P[1][1] -= K_1 * t_1;
angle+= K_0 * angle_err;
q_bias += K_1 * angle_err;
angle_dot = gyro_m-q_bias;//也许应该用last_angle-angle
}

//-----------------------
void  PIDD(){
        kp=1;ki=1;kd=1;kpp=1;
        aaxdot=0.5*aaxdot+0.5*angle_dot;
        aax=0.5*aax+0.5*angle;
        position_dot=Output*0.04;                //利用PWM值代替轮速传感器的信号
        position_dot_filter*=0.9;                //车轮速度滤波
        position_dot_filter+=position_dot*0.1;        
        positiono+=position_dot_filter;              
  positiono=constrain(positiono,-500,500);
    Output= 80*aax *kp + 1*aaxdot *ki +0.1*positiono *kd + 1*position_dot_filter *kpp;
    //if(LLOutput==0)
    //{LOutput=Output-abs(Output);
    // ROutput=Output-abs(out;
// }else
    LOutput=Output+LLOutput;
    ROutput=Output-RROutput;
    //Output= K_angle*angle *kp + K_angle_dot*angle_dot *ki +K_position*positiono *kd + K_position_dot*position_dot_filter *kpp;
}
//-------------电机正反转 PWM输出-----------
void PWMB(){
  if(LOutput<0)
{
     digitalWrite(TN1, LOW);//left back
     digitalWrite(TN2, HIGH);//left former
     //digitalWrite(TN3, LOW);//right back
     //digitalWrite(TN4, HIGH); //right former
}
else if(LOutput>0)
{
     digitalWrite(TN1, HIGH);//left back
     digitalWrite(TN2, LOW);//left former
    // digitalWrite(TN3, HIGH);//right back
    // digitalWrite(TN4, HIGH); //right former
}
else
{
    digitalWrite(TN1, HIGH);//left back
     digitalWrite(TN2, HIGH);//left former
     //digitalWrite(TN3, HIGH);//right back
   // digitalWrite(TN4, HIGH); //right former
}
//-----------------------------------------
if(ROutput<0)
{
     digitalWrite(TN3, LOW);//right back
     digitalWrite(TN4, HIGH); //right former
}
else if(ROutput>0)
{
    digitalWrite(TN3, HIGH);//right back
    digitalWrite(TN4, LOW); //right former
}
else
{
   digitalWrite(TN3, HIGH);//right back
   digitalWrite(TN4, HIGH); //right former
}
    analogWrite(ENA,min(255,abs(LOutput)+45)); //PWM调速a==0-255
    analogWrite(ENB,min(255,abs(ROutput)+45));
}
回复

使用道具 举报

发表于 2013-3-23 21:49:02 | 显示全部楼层
能不能给加点注释...水平差,没注释不好看....楼主见谅
回复 支持 反对

使用道具 举报

发表于 2013-3-23 21:55:58 | 显示全部楼层


楼住车子可以站起来吗?


1.可以将这2行取消了
aaxdot=0.5*aaxdot+0.5*angle_dot;
        aax=0.5*aax+0.5*angle;

2.测试 发送和接收器能否正常工作
回复 支持 反对

使用道具 举报

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

本版积分规则 需要先绑定手机号

Archiver|联系我们|极客工坊

GMT+8, 2024-4-29 08:44 , Processed in 0.040060 second(s), 19 queries .

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2021, Tencent Cloud.

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