找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 5287|回复: 3
打印 上一主题 下一主题
收起左侧

单片机PID算法控制直流电机转速程序,可稳定调节到目标转速

  [复制链接]
跳转到指定楼层
楼主
根据实际转速来稳定调节到目标转速


单片机源程序如下:
  1. #include<reg52.h>
  2. #include<stdio.h>
  3. #define uchar unsigned char
  4. #define uint unsigned int
  5. #define THC0 0xf9
  6. #define TLC0 0x0f   //2ms
  7. unsigned char  code Duan[]={0x3F, 0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F};//共阴极数码管,0-9段码表
  8. unsigned char  Data_Buffer[8]={0,0,0,0,0,0,0,0};

  9. uchar i=0;
  10. sbit AddSpeed=P1^1;
  11. sbit SubSpeed=P1^2;
  12. sbit PWM_FC=P1^0;
  13. int e ,e1 ,e2 ;//pid 偏差
  14. float uk ,uk1 ,duk ;//pid输出值
  15. float Kp=10,Ki=12,Kd=1.6;//pid控制系数        10,12,1.5
  16. int out=0;
  17. uint SpeedSet=380;
  18. uint cnt=0;
  19. uint Inpluse=0,num=0;//脉冲计数
  20. uint PWMTime=100;//脉冲宽度
  21. unsigned char  arry[];
  22. void SendString(uint ch);
  23. void PIDControl();
  24. void SystemInit();
  25. void delay(uchar x);
  26. void PWMOUT();
  27. void SetSpeed();
  28. void SegRefre();
  29. /**************主函数************/
  30. void main()
  31. {
  32.         SystemInit();
  33.         while(1)
  34.         {
  35.                 SetSpeed();
  36.                 SegRefre();
  37.                 PWMOUT();

  38.         }
  39. }

  40. void PIDControl()        //pid偏差计算
  41. {
  42.         e=SpeedSet-num;
  43.         duk=(Kp*(e-e1)+Ki*e+Kd*(e-2*e1+e2))/50;         //+Kd*(e-2e1+e2)
  44.         uk=uk1+duk;
  45.         out=(int)uk;
  46.         if(out>1000)
  47.         {
  48.                 out=1000;
  49.         }
  50.         else if(out<0)
  51.         {
  52.                 out=0;
  53.         }
  54.         uk1=uk;
  55.         e2=e1;
  56.         e1=e;
  57.         PWMTime=out;
  58. }

  59. void delay(uchar x)
  60. {
  61.         uint i,j;
  62.         for(i=x;i>0;i--)
  63.                 for(j=50;j>0;j--);
  64. }

  65. void PWMOUT()
  66. {
  67.         if(cnt<PWMTime)
  68.         {
  69.                 PWM_FC=1;
  70.         }
  71.         else
  72.         {
  73.                 PWM_FC=0;
  74.         }
  75.         if(cnt>1000) cnt=0;
  76. }
  77. void SystemInit()
  78. {
  79.         TMOD=0X21;   
  80.         TH0=THC0;
  81.         TL0=TLC0;
  82.         TH1=0xC0;
  83.         TL1=0XC0;
  84.         ET1=1;
  85.         ET0=1;
  86.         TR0=1;
  87.         TR1=1;
  88.         EX0=1;     //中断0用来测量转速
  89.         IT0=1;
  90.         EA=1;
  91.         e =0;
  92.         e1=0;
  93.         e2=0;
  94. }
  95. void SetSpeed()
  96. {
  97.         if(AddSpeed==0)
  98.         {
  99.                 delay(200);
  100.                 if(AddSpeed==0)
  101.                 {
  102.                         SpeedSet+=10;
  103.                         if(SpeedSet>1500)
  104.                         {
  105.                                 SpeedSet=1500;
  106.                         }
  107.                 }
  108.         }
  109.         if(SubSpeed==0)
  110.         {
  111.                 delay(200);
  112.                 if(SubSpeed==0)
  113.                 {
  114.                         SpeedSet-=10;
  115.                         if(SpeedSet<0) SpeedSet=0;
  116.                 }
  117.         }
  118. }
  119. void SegRefre()                  //显示刷新
  120. {
  121.          Data_Buffer[0]=SpeedSet/1000;
  122.          Data_Buffer[1]=SpeedSet%1000/100;
  123.          Data_Buffer[2]=SpeedSet%100/10;
  124.          Data_Buffer[3]=SpeedSet%10;
  125.          Data_Buffer[4]=num/1000;
  126.          Data_Buffer[5]=num%1000/100;
  127.          Data_Buffer[6]=num%100/10;
  128.          Data_Buffer[7]=num%10;
  129. }

  130. void int0() interrupt 0
  131. {
  132.         Inpluse++;
  133. }
  134. void t0() interrupt 1
  135. {
  136.         static unsigned char Bit=0;//静态变量,退出程序值保留
  137.         static unsigned int time=0;
  138. //        static unsigned int aa=0;
  139.         TH0=THC0;
  140.         TL0=TLC0;
  141. //        aa++;
  142. //        if(aa==50)
  143. //        {
  144. //                aa=0;
  145. ////                flag0=1;
  146. //        }

  147.         Bit++;
  148.         time++;  //转速测量周期
  149.         if(Bit>8) Bit=0;
  150.         P0=0xff;
  151.         P2=Duan[Data_Buffer[Bit]];
  152.         switch(Bit)
  153.         {
  154. ……………………

  155. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

所有资料51hei提供下载:
案列1 PID自动控制电机转速.zip (75.46 KB, 下载次数: 159)

评分

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

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏8 分享淘帖 顶1 踩
回复

使用道具 举报

沙发
ID:390416 发表于 2021-1-19 16:10 | 只看该作者
做过实物测试吗?
回复

使用道具 举报

板凳
ID:880346 发表于 2021-1-25 17:23 | 只看该作者
带负载时有处理吗
回复

使用道具 举报

地板
ID:684871 发表于 2021-3-7 20:49 来自手机 | 只看该作者
调速的模型的传递函数怎么写?
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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