标题: 单片机用PID算法控制直流电机转速 源代码+仿真 成功版 [打印本页]

作者: kk民    时间: 2017-6-4 12:35
标题: 单片机用PID算法控制直流电机转速 源代码+仿真 成功版
使用51单片机用PID算法来控制直流电机的转速.带有3个按钮 切换电机转动方向,调节p,调节转速.测试已经成功

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载):


单片机源程序如下:
  1. #include<reg52.h>
  2. #include"lcd1602.h"

  3. sfr T2MOD = 0x0c9;
  4. #define uchar unsigned char
  5. #define uint unsigned int

  6. sbit Q0 = P2^4;
  7. sbit Q1 = P2^5;
  8. sbit Q2 = P2^6;
  9. sbit Q3 = P2^7;

  10. sbit PWM                 = P1^7;
  11. sbit UP                         = P1^0;
  12. sbit DOWM                 = P1^1;
  13. sbit GORB                = P2^3; //换相
  14. sbit ADDSPEED         = P1^2;
  15. sbit SUBSPEED        = P1^3;

  16. uint tuint = 65535;
  17. uint tpwm = 1;        //pwm周期为10000us tpwm变量表示pwm高电平时间,也相当于占空比 (仿真时,频率高时,电机反应慢。在实物上要加大频率)
  18. uchar t1_flag = 0;

  19. uint pulse = 0;
  20. uint t0_flag = 0;
  21. uchar t2_flag = 0;
  22. bit t2_over = 0;
  23. bit Just_Get = 1;


  24. #define         ZZ                 { Q0 = 0;Q1 = 0;Q2 = 1;Q3 = 1;}        //正转
  25. #define         FZ                 { Q0 = 1;Q1 = 1;Q2 = 0;Q3 = 0;}        //反转
  26. #define         STOP        { Q0 = 1;Q1 = 0;Q2 = 1;Q3 = 0;}        //停止
  27. //禁止出现 Q0 = 0;Q1 = 1;Q2 = 0;Q3 = 1; 不然会烧掉mos管

  28. //************************ PID *************************************
  29. float now = 0,bef = 0,bbef = 0;         //本次采样值,上次采样值,上上次采样值
  30. float err_now,err_bef,err_bbef;                //当前偏差,上次偏差,上上次偏差
  31. float error_add = 0;                                //所有偏差之和
  32. float set = 25;                                                //设定值

  33. float kp = 25;
  34. float ki = 25;
  35. float kd = 0;

  36. //*****************************************************************

  37. void delayms(uint ms)//延时?个 ms
  38. {
  39.     uchar a,b,c;
  40.         while(ms--)
  41.         {
  42.           for(c=1;c>0;c--)
  43.         for(b=142;b>0;b--)
  44.             for(a=2;a>0;a--);
  45.         }
  46. }

  47. void timer_init()
  48. {
  49.         EA = 1;
  50.         ET0 = 1;
  51.         ET1 = 1;
  52.         ET2 = 1;
  53.        
  54.         TMOD = 0x15; //定时器0 计数模式 定时器1模式1
  55.         T2MOD = 0x01;
  56.        
  57.         TH0 = TL0 = 255;
  58.         TH2 = 0x3C;
  59.         TL2 = 0xB0;                //50MS
  60.        
  61. }
  62. void timer1() interrupt 3
  63. {
  64.         if(t1_flag == 0)
  65.         {
  66.                 t1_flag = 1;
  67.                 PWM = 1;
  68.                 TH1 = (tuint - tpwm + 1)/256;
  69.                 TL1 = (tuint - tpwm + 1)%256;
  70.                
  71.         }
  72.         else
  73.         {
  74.                 t1_flag = 0;
  75.                 PWM = 0;
  76.                 TH1 = (tuint - 10000 + tpwm + 1)/256;
  77.                 TL1 = (tuint - 10000 + tpwm + 1)%256;
  78.         }
  79. }

  80. void timer0() interrupt 1
  81. {
  82.         TH0 = TL0 = 255;
  83.         t0_flag++;
  84. }
  85. void timer2() interrupt 5
  86. {
  87.         TF2 = 0;
  88.         TH2 = 0x3C;
  89.         TL2 = 0xB0;                //50MS
  90.        
  91.         t2_flag++;
  92.        
  93.         if(t2_flag == 2)
  94.         {
  95.                 TR0 = 0;
  96.                 TR2 = 0;
  97.                 t2_flag = 0;
  98.                 t2_over = 1;        //表示100ms时间到
  99.         }
  100. }
  101. void GetPulse()
  102. {
  103.         t0_flag = 0;
  104.         t2_flag = 0;
  105.        
  106.         TH0 = TL0 = 255;
  107.         TH2 = 0x3C;
  108.         TL2 = 0xB0;                //50MS
  109.        
  110.         TR0 = 1;
  111.         TR2 = 1;
  112. }

  113. int PID()        //增量式PID
  114. {
  115.         int change;

  116.         err_now = set - now;
  117.         err_bef = set - bef;
  118.         err_bbef = set - bbef;
  119.        
  120.         change = kp*(err_now - err_bef) + ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
  121.        
  122. /*       
  123.         if(set >= now)
  124.         {       
  125.                 if(set - now > 1)
  126.                         change = kp*(err_now - err_bef) + ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
  127.                 else
  128.                         change = 0.2*kp*(err_now - err_bef) + 0.5*ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
  129.         }
  130.         else if(now > set)
  131.         {
  132.                 if(now - set > 1)
  133.                         change = kp*(err_now - err_bef) + ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
  134.                 else
  135.                         change = 0.2*kp*(err_now - err_bef) + 0.5*ki*err_now + kd*(err_now - 2*err_bef + err_bbef);
  136.                        
  137.         }
  138. */
  139.        
  140.         //change = (kp + ki + kd)*(set - now) + (-kp - 2*kd)*(set - bef) + kd*(set - bbef);
  141.         //change = kp*(set - now) + ki*(set - bef) + kd*(set - bbef);
  142.         if(change > 0)
  143.         {
  144.                 printchar(1,10,'+');       
  145.                 printuint(1,11,4,change);
  146.                
  147.         }
  148.         else if(change < 0)
  149.         {       
  150.                 printchar(1,10,'-');
  151.                 printuint(1,11,4,-change);
  152.         }
  153.         else if(change == 0)
  154.         {       
  155.                 printchar(1,10,' ');
  156.                 printword(1,11," 0  ");

  157.         }
  158.        
  159.         return(change);
  160. }

  161. int PID2()                //位置式PID
  162. {
  163.        
  164.         int num = 0;
  165.         static num_bef = 0;
  166.        
  167.         err_now = set - now;
  168.         err_bef = set - bef;
  169.        
  170.         error_add = error_add + err_now;  //误差累加。一旦误差为0则error_add的值不变,PID输出值不变

  171.         num = kp*err_now + ki*error_add + kd*(err_now - err_bef);
  172.        
  173. /*       
  174.         if(set - now >= 0)
  175.         {       
  176.                 if(set - now > 1)
  177.                         num = kp*err_now + ki*error_add + kd*(err_now - err_bef);
  178.                 else
  179.                         num = 0.1*kp*err_now + ki*error_add + kd*(err_now - err_bef);
  180.         }
  181.         else
  182.         {
  183.                 if(now - set > 1)
  184.                         num = kp*err_now + ki*error_add + kd*(err_now - err_bef);
  185.                 else
  186.                         num = 0.1*kp*err_now + ki*error_add + kd*(err_now - err_bef);
  187.                        
  188.         }
  189.         */
  190.        
  191.         if(num > num_bef)
  192.         {
  193.                 printchar(1,10,'+');       
  194.                 printuint(1,11,4,num - num_bef);
  195.         }
  196.         else if(num < num_bef)
  197.         {
  198.                 printchar(1,10,'-');       
  199.                 printuint(1,11,4,num_bef - num);
  200.         }
  201.         else
  202.         {       
  203.                 printchar(1,10,' ');
  204.                 printuint(1,11,4,0);
  205.         }
  206.        
  207.         num_bef = num;
  208.        
  209.         return((uint)num);
  210. }

  211. void main()
  212. {       
  213.        
  214.         lcd_init();
  215.         timer_init();
  216.         TH1 = TL1 = 255;
  217.        
  218.         printword(0,0,"P:");                //比例系数
  219.         printword(0,5,"S:");                //设定值
  220.         printword(1,0,"TPWM:");                //当前占空比
  221.         printword(0,10,"PS:");                //当前电机反馈的每秒脉冲数
  222.        
  223.         while(1)
  224.         {
  225.                 if(GORB == 1)
  226.                 {        ZZ;                }
  227.                 else
  228.                 {        FZ;                }
  229.                
  230.                 if(ADDSPEED == 0)
  231.                         set++;
  232.                 if(SUBSPEED == 0)
  233.                         set--;
  234.                
  235.                 if(Just_Get == 1)
  236.                 {       
  237.                         Just_Get = 0;
  238.                         GetPulse();
  239.                 }
  240.                 else if(t2_over == 1)
  241.                 {       
  242.                         t2_over = 0;
  243.                         Just_Get = 1;
  244.                         pulse = t0_flag;
  245.                         bbef = bef;
  246. ……………………

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

所有资料51hei提供下载:
PID算法控制电机转速.rar (100.25 KB, 下载次数: 1239)



作者: admin    时间: 2017-6-4 16:33
好资料,51黑有你更精彩!!!
作者: qjyjack612510    时间: 2017-6-4 17:39
不懂,下来慢慢学
作者: kevin110125    时间: 2017-8-9 14:09
很好,学习一下!
作者: a77303383    时间: 2017-9-3 14:24
不错加油
作者: 别人没有的名字    时间: 2017-10-15 00:21
棒棒哒

作者: 陆地巡洋舰    时间: 2017-10-15 11:27
慢慢学习一下
作者: 1633278314wl    时间: 2017-10-15 14:21
nice   好大拇指
作者: 别人没有的名字    时间: 2017-10-15 22:22
请问lcd显示屏上的p值是什么用途呢?
作者: lhf211185    时间: 2017-10-17 01:00
努力学习学习
作者: 夏天xx    时间: 2017-10-29 15:49
P值是什么

作者: wuyanzuzzz    时间: 2017-11-21 20:55
谢谢分享    好
作者: peigang123    时间: 2017-12-17 11:50
谢谢分享谢谢分享谢谢分享
作者: peigang123    时间: 2017-12-17 11:53
谢谢分享

作者: peigang123    时间: 2017-12-17 14:23
看看
谢谢分享  
作者: sinolover    时间: 2017-12-18 10:11
正在做直流电机控制,学习下
作者: sinolover    时间: 2017-12-18 10:12
额,没币,下起不了
作者: aaaa_7777    时间: 2018-3-13 23:22

好资料,51黑有你更精彩!!!
作者: goldbr    时间: 2018-4-6 18:13
学习一下
作者: hllt    时间: 2018-4-12 11:54
很好的资料,刚好最近在做这个。
作者: coldwindhot    时间: 2018-4-19 11:04
谢谢分享

作者: 2333admin    时间: 2018-5-7 22:01
谢谢分享
作者: zh_silence    时间: 2018-5-10 10:35
非常感谢,新手入门,很有用
作者: HEARD    时间: 2018-5-23 14:24
谢谢楼主
作者: zhanggaochina    时间: 2018-6-6 18:47
不错,值得学习
作者: combathyl    时间: 2018-6-6 21:09
这个源代码直接放到keil里面编译,是有警告的啊?
作者: q86346643    时间: 2018-6-21 17:04
谢谢哈哈
作者: 电子菜鸟啊    时间: 2018-6-25 21:12
下来慢慢学

作者: 小阿哈    时间: 2018-7-13 10:53
pid入坑中
作者: 123feng    时间: 2018-7-13 11:55
谢谢分享
作者: lm1720755472    时间: 2018-7-13 17:06
非常好
作者: 只为学习到一点    时间: 2018-7-14 15:09
值得学习
作者: 23932209    时间: 2018-8-19 07:49
很想试试看
作者: cjztiger    时间: 2018-9-14 17:14
很想试试看
作者: kikko    时间: 2018-10-10 17:43
感谢大佬
作者: xxz1061018609    时间: 2018-10-13 09:46
第一次用PID算法来控制,文件里位置式和增量式都涉及到了,很详细
作者: linyou    时间: 2018-11-24 11:50
正好,试着做做

作者: brycew    时间: 2018-12-22 21:47
谢谢楼主分享
作者: wlee    时间: 2018-12-22 22:24
努力学习亚
作者: jmy11    时间: 2018-12-22 22:28
好帖子,试试做
作者: 杨柳青    时间: 2019-1-4 21:51
厉害!
作者: xode    时间: 2019-1-4 23:51
谢谢分享
作者: 伯爵默默    时间: 2019-1-12 14:55
东西很不错
作者: nipingbo    时间: 2019-2-11 13:28
东西很不错

作者: nipingbo    时间: 2019-2-11 13:28
东西很不错

作者: 1172047001    时间: 2019-2-12 13:06
很给力!
作者: 648025666    时间: 2019-3-19 18:08
非常感谢,新手入门
作者: lyy001    时间: 2019-3-29 15:34
好资料,51黑有你更精彩!!!
作者: lyy001    时间: 2019-3-29 16:51
第一次用PID算法来控制,文件里位置式和增量式都涉及到了,很详细
作者: yll1995    时间: 2019-3-29 18:11
很好,谢谢楼主
作者: 923975623    时间: 2019-4-22 19:36
赞 很有用
作者: 蝶舞狂雷    时间: 2019-5-5 09:34
谢谢分享
作者: qxl1997119    时间: 2019-5-5 12:30
棒棒哒
作者: 13327291097    时间: 2019-5-5 14:42
好资料,51黑有你更精彩!!!
作者: machenshuo    时间: 2019-5-6 17:48
从图上看,电机和单片机的输入电压一样,可以不一样吗?
作者: machenshuo    时间: 2019-5-6 20:30
附件在哪??
作者: 胡旭飞飞飞飞    时间: 2019-5-9 16:31
这个让我学到了很多!
作者: timelesszn    时间: 2019-5-24 14:13
很好,很牛逼啊
作者: 2632960016    时间: 2019-5-31 11:14
新手入门先保存下来

作者: Jazz-HipHop    时间: 2019-6-24 13:10
正在做这个,感谢
作者: 825673316    时间: 2019-6-27 22:45
非常期待,希望是好的
作者: 陈小晓    时间: 2019-7-29 20:54
注释也太少了点吧,理解起来有点困难
作者: hht805656153    时间: 2019-7-30 17:49
好资料,51黑有你更精彩!!!
作者: sym111    时间: 2019-7-31 08:00

谢谢分享
作者: 花名张清    时间: 2019-7-31 09:16
好资料,51黑有你更精彩!!!
作者: meng666meng    时间: 2019-8-2 11:08
很不错的贴,可是想下载积分不够
作者: 小萝卜头    时间: 2019-8-2 16:26
给力啊,今天刚学到PID
作者: china043    时间: 2019-8-5 10:23
不错加油
作者: zhangsl168    时间: 2019-8-5 16:56
真是高人很多啊,原来这个也可做啊!
作者: JTR2017    时间: 2019-9-18 16:11
灰常感谢
作者: 刘继智    时间: 2019-10-3 15:58
想要学知识,麻烦给一个吧
作者: mmmaaaxxx    时间: 2019-11-12 18:15
谢谢分享    好
作者: 阿斯顿发生    时间: 2019-11-14 19:04
赞一个!
作者: hj08102    时间: 2020-2-28 12:11
很详细的代码,下载实验下,谢谢楼主分享
作者: yoghourt    时间: 2020-2-28 14:38
非常实用,谢谢
作者: sinolover    时间: 2020-3-17 17:03
我自己弄的,一直不理想
作者: 15649862501    时间: 2020-3-22 17:05
谢谢分享
作者: 2794446011    时间: 2020-4-2 14:47
先Mark
作者: sance    时间: 2020-4-25 09:16
感谢分享 感谢
作者: 15245340616    时间: 2020-6-26 17:57
谢谢分享
作者: Jeff_BlindCat    时间: 2020-6-26 23:20
好资料,谢谢楼主分享。
作者: ygw911651    时间: 2020-7-27 19:33
proteus的文件没有啊
作者: 周11    时间: 2020-8-11 23:37
这个是用52的板子吗
作者: fcc521    时间: 2020-8-12 14:30
pid里的printchar是自定义的吗
作者: adsdasfewfwe    时间: 2020-8-16 17:27
感谢分享,太给力了,这个好难,还好有你
作者: asuka鸟    时间: 2020-12-1 15:15

感谢分享,太给力了,这个好难,还好有你
作者: asuka鸟    时间: 2020-12-2 08:58
讲的很棒
作者: 小啊嘉    时间: 2024-4-26 01:20

nice   好大拇指
作者: 徐久博    时间: 2024-4-26 19:50
我下载了·准备安装·谢谢分享
作者: ffqx@!68    时间: 2024-4-28 08:56
看了,真的实用!
作者: haide1998    时间: 2024-5-2 20:37
谢谢分享,学习了!
作者: 随风飘远    时间: 2024-5-11 10:02
电机驱动是不是需要滤波电路? 有谁了解吗?
作者: 51heiartist    时间: 2024-7-20 15:10
是有刷的吗
作者: sdfbhbk111    时间: 2024-10-24 08:58
学到了很多知识
作者: cuijb    时间: 2024-10-27 11:00
下管驱动应该没问题,上管驱动能用吗
作者: jovew    时间: 2024-10-28 13:46
这个程序驱动的电机是步进电机吗?
作者: xdc2005    时间: 2025-6-19 16:18
很给力
作者: xdc2005    时间: 2025-6-20 18:06
谢谢分享,仿真确实能成功启动





欢迎光临 (http://www.51hei.com/bbs/) Powered by Discuz! X3.1