找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1927|回复: 1
收起左侧

单片机PID转速测量控制程序 包括Proteus仿真

  [复制链接]
ID:634507 发表于 2019-11-30 15:01 | 显示全部楼层 |阅读模式
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
51hei.png

单片机源程序如下:
  1. #include <REGX51.H>
  2. #include<stdio.h>
  3. #define uchar unsigned char
  4. #define uint unsigned int
  5. #define THC0 0xf8
  6. #define TLC0 0x50   //2ms,0x30,含中断处理时间时,0x50
  7. #include "LCD.H"
  8. #include "delay.h"

  9. unsigned char aa[]={'T','a','r','g','e','t',' ',' ',' ',' ',' ','r','/','m','i','n'};  //目标转速。Target            r/min
  10. unsigned char cc[]={'A','c','t','u','a','l',' ',' ',' ',' ',' ','r','/','m','i','n'};           //实测转速: Actual      r/min
  11.                                                                                                                                                                                                                                  
  12. uchar i=0;
  13. sbit k1=P1^3;
  14. sbit k2=P1^4;
  15. sbit k3=P1^5;
  16. sbit k4=P1^6;
  17. sbit k5=P1^7;
  18. sbit PWM_FC=P3^5;
  19. sbit IN1=P3^4;
  20. sbit IN2=P3^3;

  21. int e ,e1 ,e2 ;//pid 偏差
  22. float uk ,uk1 ,duk ;//pid输出值
  23. float Kp=0.25,Ki=0.05,Kd=0.015;//pid控制系数        0.1 0.05 0.016

  24. /*
  25. PID的参数设置可以参照以下来进行:  
  26.      参数整定找最佳,从小到大顺序查;  
  27.          先是比例后积分,最后再把微分加;  
  28.          曲线振荡很频繁,比例度盘要放大;  
  29.          曲线漂浮绕大湾,比例度盘往小扳;  
  30.          曲线偏离回复慢,积分时间往下降;  
  31.          曲线波动周期长,积分时间再加长;  
  32.          曲线振荡频率快,先把微分降下来;  
  33.          动差大来波动慢。微分时间应加长;  
  34.          理想曲线两个波,前高后低4比1 ; 
  35.          一看二调多分析,调节质量不会低;
  36. */

  37. int out=0;
  38. uint SpeedSet=3000;
  39. uint cnt=0;
  40. uint Inpluse=0,num=0,zs;//脉冲计数
  41. uint PWMTime=100;//脉冲宽度
  42. void PIDControl();
  43. void SystemInit();
  44. void delay(uchar x);
  45. void PWMOUT();
  46. void SetSpeed();

  47. /**************主函数************/
  48. void main()
  49. {
  50.         SystemInit();
  51.          init();
  52.          LCD_Write_String(0,0,aa);
  53.         zs=1;
  54.         while(1)
  55.         {
  56.                 SetSpeed();
  57.                 if(zs==1)
  58.                 {
  59.                     zs=0;
  60.                         cc[7]=num/1000+'0';
  61.                         cc[8]=num/100%10+'0';
  62.                         cc[9]=num/10%10+'0';
  63.                         cc[10]=num%10+'0';
  64.                         LCD_Write_String(0,1,cc);
  65.                 }
  66.         }
  67. }

  68. void PIDControl()        //pid偏差计算
  69. {
  70.         e=SpeedSet-num;
  71.         duk=(Kp*(e-e1)+Ki*e+Kd*(e-2*e1+e2));  
  72.         uk=uk1+duk;
  73.         out=(int)uk;
  74.         if(out>1000)
  75.         {
  76.                 out=1000;
  77.         }
  78.         else if(out<0)
  79.         {
  80.                 out=0;
  81.         }
  82.         uk1=uk;
  83.         e2=e1;
  84.         e1=e;
  85.         PWMTime=out;
  86. }

  87. void delay(uchar x)
  88. {
  89.         uint i,j;
  90.         for(i=x;i>0;i--)
  91.                 for(j=50;j>0;j--);
  92. }

  93. void PWMOUT()
  94. {
  95.         if(cnt<PWMTime)
  96.         {
  97.                 PWM_FC=1;
  98.         }
  99.         else
  100.         {
  101.                 PWM_FC=0;
  102.         }
  103.         if(cnt>1000) cnt=0;
  104. }
  105. void SystemInit()
  106. {
  107.         TMOD=0X21;    //t1用来串口t2定时
  108.         TH0=THC0;
  109.         TL0=TLC0;
  110.         TH1=0xC0;
  111.         TL1=0XC0;
  112.         ET1=1;
  113.         ET0=1;
  114.         TR0=1;
  115.         TR1=1;
  116.         EX0=1;     //中断0用来测量转速
  117.         IT0=1;
  118.         EA=1;
  119.         e =0;
  120.         e1=0;
  121.         e2=0;
  122.         IN1=1;
  123.         IN2=1;
  124. }
  125. void SetSpeed()
  126. {
  127.         if(k1==0)
  128.         {
  129.                 delay(100);
  130.                 if(k1==0)
  131.                 {
  132.                    IN1=0;
  133.                    IN2=1;
  134.                 }
  135.         }
  136.         if(k2==0)
  137.         {
  138.                 delay(100);
  139.                 if(k2==0)
  140.                 {
  141.                    IN1=1;
  142.                    IN2=1;
  143.                 }
  144.         }
  145.         if(k3==0)
  146.         {
  147.                 delay(100);
  148.                 if(k3==0)
  149.                 {
  150.                    IN1=~IN1;
  151.                    IN2=~IN2;
  152.                 }
  153.                 while(k3==0);
  154.         }
  155.         if(k4==0)
  156.         {
  157.                 delay(100);
  158.                 if(k4==0)
  159.                 {
  160.                         SpeedSet+=10;
  161.                         if(SpeedSet>3500)
  162.                         {
  163.                                 SpeedSet=3500;
  164.                         }
  165.                 }
  166.         }
  167.         if(k5==0)
  168.         {
  169.                 delay(100);
  170.                 if(k5==0)
  171.                 {
  172.                         SpeedSet-=10;
  173.                         if(SpeedSet<0) SpeedSet=0;
  174.                 }
  175.         }        
  176.         aa[7]=SpeedSet/1000+'0';
  177.         aa[8]=SpeedSet/100%10+'0';
  178.         aa[9]=SpeedSet/10%10+'0';
  179.         aa[10]=SpeedSet%10+'0';
  180.         LCD_Write_String(0,0,aa);
  181. }


  182. void int0() interrupt 0
  183. {
  184.         Inpluse++;
  185. }
  186. void t0() interrupt 1
  187. {
  188.         static unsigned int time=0;

  189.         TH0=THC0;
  190.         TL0=TLC0;
  191.         time++;  //转速测量周期
  192.         if(time>500)
  193.         {
  194.                 zs=1;
  195.                 time=0;
  196.                 num=Inpluse;         //计算式中是仿真时,码盘数60时的情况,如果码盘数n=10时,num=Inpluse*60/n=Inpluse*6;
  197.                 Inpluse=0;
  198.                 PIDControl();
  199.         }
  200.         PWMOUT();
  201. }
  202. void timer_1()  interrupt 3
  203. {
  204.            cnt++;        //cnt越大占空比越高2.5Khz
  205. }


复制代码
51hei.png

全部资料51hei下载地址:
2-PID转速测量控制_LCD.zip (296.09 KB, 下载次数: 120)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:693716 发表于 2020-3-9 11:39 | 显示全部楼层
这个仿真有点意思,学习了下,主要学习pid控制
回复

使用道具 举报

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

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

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