1

关于投票
PID调节经验总结

PID控制器参数选择的方法很多,例如试凑法、临界比例度法、扩充临界比例度法等。但是,对于PID控制而言,参数的选择始终是一件非常烦杂的工作,需要经过不断的调整才能得到较为满意的控制效果。依据经验,一般PID参数确定的步骤如下[42]:
(1)        确定比例系数Kp
确定比例系数Kp时,首先去掉PID的积分项和微分项,可以令Ti=0、Td=0,使之成为
纯比例调节。输入设定为系统允许输出最大值的60%~70%,比例系数Kp由0开始逐渐增大,直至系统出现振荡;再反过来,从此时的比例系数Kp逐渐减小,直至系统振荡消失。记录此时的比例系数Kp,设定PID的比例系数Kp为当前值的60%~70%。
(2)        确定积分时间常数Ti
比例系数Kp确定之后,设定一个较大的积分时间常数Ti,然后逐渐减小Ti,直至系统出现振荡,然后再反过来,逐渐增大Ti,直至系统振荡消失。记录此时的Ti,设定PID的积分时间常数Ti为当前值的150%~180%。
(3)        确定微分时间常数Td
微分时间常数Td一般不用设定,为0即可,此时PID调节转换为PI调节。如果需要设定,则与确定Kp的方法相同,取不振荡时其值的30%。
(4)        系统空载、带载联调
对PID参数进行微调,直到满足性能要求。
本文来自: DSP交流网(www.hellodsp.com) 详细出处参考:http://www.hellodsp.com/bbs/viewthread.php?tid=688

 

 

PID代码

//定义变量
float Kp;                       //PI调节的比例常数
float Ti;                       //PI调节的积分常数
float T;                        //采样周期
float Ki;
float ek;                       //偏差e[k]
float ek1;                      //偏差e[k-1]
float ek2;                      //偏差e[k-2]
float uk;                       //u[k]
signed int uk1;                 //对u[k]四舍五入取整
signed int adjust;              //调节器输出调整量

//变量初始化
    Kp="4";
    Ti="0"。005;
    T="0".001;
// Ki="KpT/Ti"=0.8,微分系数Kd=KpTd/T=0.8,Td=0.0002,根据实验调得的结果确定这些参数
    ek="0";
    ek1=0;
    ek2=0;
    uk="0";
    uk1=0;
    adjust="0";

int piadjust(float ek)  //PI调节算法
{
    if( gabs(ek)<0.1 )
    {
       adjust="0";
    }
    else
    {    
       uk="Kp"*(ek-ek1)+Ki*ek;  //计算控制增量
       ek1=ek;
      
       uk1=(signed int)uk;
       if(uk>0)
       {
          if(uk-uk1>=0.5)
          {
             uk1=uk1+1;
          }
       }
       if(uk<0)
       {
          if(uk1-uk>=0.5)
          {
             uk1=uk1-1;
          }
       }
       adjust="uk1";   
    }       
   
   
    return adjust;
}

下面是在AD中断程序中调用的代码。
      
       。。。。。。。。。。。
       else //退出软启动后,PID调节,20ms调节一次
           {
              EvaRegs.CMPR3=EvaRegs.CMPR3+piadjust(ek);//误差较小PID调节稳住
              if(EvaRegs.CMPR3>=890)
              {
                 EvaRegs.CMPR3=890; //限制PWM占空比
              }       
           }

本文来自: DSP交流网(www.hellodsp.com) 详细出处参考:http://www.hellodsp.com/bbs/viewthread.php?tid=688

http://www.intelligentcar.cn/ShowPost.asp?ThreadID=516

系统分类: 单片机
用户分类: 智能车
标签: 无标签
来源: 整理
发表评论 阅读全文(279) | 回复(0)

0

关于投票
CodeWarrior高级教程
  网上讲CodeWarrior的pdf倒是不少,可是都是入门级的,只教你怎么创建工程,对于想更熟练使用这个工具的人来说是远远不够的。一次偶然的机会在网上找到了个高级点的,里面讲的有prm文件的使用,中断函数的编写,我想很多人在刚接触CodeWarrior时都弄不清楚这里面的文件是怎么回事,和其他IDE有什么区别,这里的C方言与标准C的小区别。看了这个你会更加清楚CodeWarrior的用法。

  这个pdf以8位飞思卡尔单片机为例,但用法与16位是相通的。

  感谢康桥人,我是从他博客上找到的,不知道他为什么不把他发布在这里?难道是觉得这东西太好,舍不得?(开玩笑的,别生气,呵呵!)




FSL 08系列单片机开发及C语言编程简介.pdf

http://download.intelligentcar.cn/bbsupfile//upfile/FSL%2008系列单片机开发及C语言编程简介.pdf

http://www.intelligentcar.cn/ShowPost.asp?ThreadID=463

系统分类: 单片机
用户分类: 智能车
标签: 无标签
来源: 整理
发表评论 阅读全文(226) | 回复(0)

1

关于投票
CCD图像采集解决方案

       近几天看到智能车论坛里有很多网友遇到CCD图像采集的麻烦,我在最开始的时候也为这个烦恼过,由于本人比较菜,在度过大概半个月的绝望日夜后,在刚准备放弃时突然发现我已经采集到正确的图像了。特再次分享,希望能解决大家当前遇到的麻烦。

      在采集图像之前,我们首先要知道摄像头输出信号的特性。目前的模拟摄像头一般都是PAL制式的,输出的信号由复合同步信号,复合消隐信号和视频信号。其中的视频信号才是真正的图像信号,对于黑白摄像头,图像越黑,电压越低,图像越白,电压越高。而复合同步信号是控制电视机的电子枪对电子的偏转的,复合消隐信号是在图像换行和换场时电子枪回扫时不发射电子。由于人眼看到的图像大于等于24Hz时人才不会觉得图像闪烁,所以PAL制式输出的图像是25Hz,即每秒钟有25幅画面,说的专业点就是每秒25帧,其中每一帧有625行。但由于在早期电子技术还不发达时,电源不稳定,容易对电视信号进行干扰,而交流电源是50Hz所以,为了和电网兼容,同时由于25Hz时图像不稳定,所以后来工程师们把一副图像分成两场显示,对于一幅画面,一共有625行,但是电子枪先扫描奇数场1,3,5.....,然后再扫描2,4,6.....,所以这样的话,一副图像就变成了隔行扫描,每秒钟就有50场了。其中具体的细节请参考这个网站

电视原理与系统

http://courseware.ecnudec.com/zsb/zjx/zjx09/zjx090000.htm

只用看前面的黑白全电视信号和PAL制式就可以了(当然如果感兴趣可以全部看完)。

 

     通过上面的内容如果你对PAL制式信号了然于心,那么就可以开始图像的采集了,PAL输出的信号有复合同步信号,复合消隐信号和视频信号。那么我们首先就是要从这三种信号中分理出复合同步信号,复合消隐信号和视频信号,以便我们对AD采样到的值进行存储,从而形成一幅画面。具体如何分离,我们使用的是LM1881视频同步分离器件,具体的硬件连接请参看论坛内相关文章(论坛里有介绍LM1881的文章,自己搜吧,我不重复了)。

   分离出行场同步,奇偶场信号后,就把他们接到单片机的外部中断口,产生中断,在中断服务程序中对AD采集到的数据进行图像存储,从而形成一个二维数组的数字图像。

   下面就说说图像采集方案,方法有多种,但我使用的方案是在行终端中读取AD采样的灰度值,在场同步中交换图像采集和处理缓存指针,并对图像进行处理,然后控制小车,在主函数中只有初始化和键盘扫描和串口输出函数。这样做效率比较高,而且可以把调试和图像采集处理分开,变成起来比较方便。

   大家遇到的还有一个很棘手的问题可能是AD采样频率该设置多大呢?建议大家先通过PLLL超频,然后把AD时钟频率设置的高点才行。

下面就把我的代码贴给大家看看吧。

//*************************************************************************
//*    *************************锁相环初始化***********************    *
//*************************************************************************
void vPLLInit(void)
{                                  //BUS-CLOCK=PLL-CLOCK/2=32M
   REFDV = 1;  // set the REFDV register 16M*2*(3+1)/(1+1)=64M
   SYNR =3;    // set the SYNR register to give us a 64 MHz PLL-clock.
    asm nop    // nops required for PLL stability.
    asm nop
    asm nop
    asm nop
   while ((CRGFLG&0x08)==0); // wait here till the PLL is locked.
   CLKSEL|=0x80;             // switch the bus clock to the PLL.
}
设置总线时钟为32M

 

//*************************************************************************
//*    *************************定时器初始化***********************    *
//*************************************************************************
void vECTInit(void)
{
  TIOS =0x00;    //设为输入捕捉
  TSCR1=0x80;    //定时器使能
  TSCR2=0x83;    //允许定时器溢出中断,定时器时钟32M/(2^3)=4M
  TCTL4=0xAA;    //触发电平:下降沿
  TIE  =0x07;    //开中断
  TFLG1=0xFF;    //清除中断标志
}

 

输入捕捉的1,2通道接行场中断。

 

//*************************************************************************
//*    ***************************AD转换初始化程序****************    *
//*************************************************************************
void vADInit(void)
{
//********************************ATD1设置*********************************
//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<控制寄存器2>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
//上电,标志位快速清零,忽略外部触发,执行一次停止,中断禁止。
  ATD1CTL2  = (ATD1CTL2_AFFC_MASK | ATD1CTL2_ADPU_MASK);
     
//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<控制寄存器3>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
//转换序列长度为1,FIFO模式,Freeze模式下继续转换。|ATD0CTL3_FIFO_MASK
  ATD1CTL3  = (ATD1CTL3_S1C_MASK);
    
//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<控制寄存器4>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
//8位精度,2AD采样周期,采样长度8。
//ATDClock=[BusClock*0.5]/[PRS+1]  ; PRS="15", divider="32"
  ATD1CTL4  =(ATD1CTL4_SRES8_MASK|ATD1CTL4_PRS0_MASK);
   
//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<控制寄存器5>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
//右对齐无符号,扫描模式连续采样,单通道采样//多通道采样|ATD0CTL5_MULT_MASK。
  ATD1CTL5  = (ATD1CTL5_DJM_MASK|ATD1CTL5_SCAN_MASK);
 
//<<<<<<<<<<<<<<<<<<<<<<<<<<禁止数字输入缓冲>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
 ATD1DIEN=0x00;
}

ATD1的0通道用于AD转换

 

下面是真正的图像采集程序

//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>   
//当前采样图像的行和列。
unsigned int ui_SampleRow=0,ui_SampleColumn=0;

//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>   
//图像数据缓存
unsigned char uca_Buffer1[IMAGE_ROW][IMAGE_COLUMN];
unsigned char uca_Buffer2[IMAGE_ROW][IMAGE_COLUMN];

//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>   
//指向当前采集数据采样缓存首地址的指针
unsigned char *puca_BufferSample=&uca_Buffer1[0][0];

//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>   
//指向当前处理数据采样缓存首地址的指针
unsigned char *puca_BufferProcess=&uca_Buffer2[0][0];

//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>   
//用于图像采集和处理交换缓存。
//(注意:在每次交换指针后保证puca_BufferTemp与puca_BufferSample相同)
unsigned char *puca_BufferTemp=&uca_Buffer1[0][0];

#pragma CODE_SEG NON_BANKED
//*************************************************************************
//*                                                *
//*    *******************输入捕捉2通道中断函数********************    *
//*                                                *
//*************************************************************************
//行同步 ,用于数据采集。
void interrupt 10 vIC2ISR(void)  
{
  unsigned char ucTemp;
  unsigned char *pucTemp;
  TFLG1_C2F=1;
 
  if(ui_SampleRow>=SAMP_ROW_START&&ui_SampleRow<SAMP_ROW_MAX)
  {
    if(ui_SampleRow%SAMP_ROW_SEP==0)
    {
      for(ui_SampleColumn=0;ui_SampleColumn<SAMP_COL_MAX;ui_SampleColumn++)
      {
        while(!ATD1STAT1_CCF0);
        if(ui_SampleColumn>=SAMP_COL_START)
        {
          if(ui_SampleColumn%SAMP_COL_SEP==0)
          {
            pucTemp="puca"_BufferSample
              +((ui_SampleRow-SAMP_ROW_START)/SAMP_ROW_SEP)*IMAGE_COLUMN
              +(ui_SampleColumn-SAMP_COL_START)/SAMP_COL_SEP;
            *pucTemp=ATD1DR0L;
          }
        }
      }
    }
  }
  ucTemp="ATD1DR0L";
  ui_SampleRow++;               //采样行坐标加一。
}
//*************************************************************************
//*                                                *
//*    *******************输入捕捉1通道中断函数********************    *
//*                                                *
//*************************************************************************
//场同步,交换缓存以及图像处理和模型车控制。
void interrupt 9 vIC1ISR(void)   
{    
  TFLG1_C1F=1;
  ui_SampleRow=0;               //把采样行坐标清零。
  ui_SampleColumn=0;
    
//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>   
//交换图像采集和处理缓存
  puca_BufferSample=puca_BufferProcess;
  puca_BufferProcess=puca_BufferTemp;
  puca_BufferTemp=puca_BufferSample;
 
//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
//系统时间加一。
  ul_SystemTime+=1;
 
//<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>   
//开中断,允许行信号中断进行采样。
  EnableInterrupts;
   
  if(uc_CarState==STATE_START)
  {
//    PORTB_BIT1=1;
   //分析图像,获取路径参数,根据路径参数控制模型车。。
    vImageProcess();
   
    //根据路径参数控制模型车。
    vAutoControl();
//    PORTB_BIT1=0;
  }

}
 


下面是我的小车程序,其中AD,输入捕捉和PLL初始化函数在Drives.c中,图像采集在Interrupts.c中。
SmartCarII_5_1.rar

系统分类: 单片机
用户分类: 智能车
标签: 无标签
来源: 原创
发表评论 阅读全文(151) | 回复(0)
总共 , 当前 /