久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 48429|回復: 95
收起左側

單片機用PID算法控制直流電機轉速 源代碼+仿真 成功版

  [復制鏈接]
ID:207635 發表于 2017-6-4 12:35 | 顯示全部樓層 |閱讀模式
使用51單片機用PID算法來控制直流電機的轉速.帶有3個按鈕 切換電機轉動方向,調節p,調節轉速.測試已經成功

仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載):
0.png 0.png 0.png

單片機源程序如下:
  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)


評分

參與人數 5黑幣 +43 收起 理由
dww465757120 + 30 很給力!
我請你x + 2 贊一個!
stu + 1 很給力!
lm1720755472 + 5 學習到知識
杜輔翼 + 5 很給力!

查看全部評分

回復

使用道具 舉報

ID:1 發表于 2017-6-4 16:33 | 顯示全部樓層
好資料,51黑有你更精彩!!!
回復

使用道具 舉報

ID:177855 發表于 2017-6-4 17:39 | 顯示全部樓層
不懂,下來慢慢學
回復

使用道具 舉報

ID:226086 發表于 2017-8-9 14:09 | 顯示全部樓層
很好,學習一下!
回復

使用道具 舉報

ID:122686 發表于 2017-9-3 14:24 | 顯示全部樓層
不錯加油
回復

使用道具 舉報

ID:239382 發表于 2017-10-15 00:21 | 顯示全部樓層
棒棒噠
回復

使用道具 舉報

ID:210141 發表于 2017-10-15 11:27 來自手機 | 顯示全部樓層
慢慢學習一下
回復

使用道具 舉報

ID:239623 發表于 2017-10-15 14:21 | 顯示全部樓層
nice   好大拇指
回復

使用道具 舉報

ID:239382 發表于 2017-10-15 22:22 | 顯示全部樓層
請問lcd顯示屏上的p值是什么用途呢?
回復

使用道具 舉報

ID:125901 發表于 2017-10-17 01:00 來自手機 | 顯示全部樓層
努力學習學習
回復

使用道具 舉報

ID:243861 發表于 2017-10-29 15:49 | 顯示全部樓層
P值是什么
回復

使用道具 舉報

ID:252056 發表于 2017-11-21 20:55 | 顯示全部樓層
謝謝分享    好
回復

使用道具 舉報

ID:262605 發表于 2017-12-17 11:50 | 顯示全部樓層
謝謝分享謝謝分享謝謝分享
回復

使用道具 舉報

ID:262605 發表于 2017-12-17 11:53 | 顯示全部樓層
謝謝分享
回復

使用道具 舉報

ID:262605 發表于 2017-12-17 14:23 | 顯示全部樓層
看看
謝謝分享  
回復

使用道具 舉報

ID:262988 發表于 2017-12-18 10:11 | 顯示全部樓層
正在做直流電機控制,學習下
回復

使用道具 舉報

ID:262988 發表于 2017-12-18 10:12 | 顯示全部樓層
額,沒幣,下起不了
回復

使用道具 舉報

ID:44165 發表于 2018-3-13 23:22 | 顯示全部樓層

好資料,51黑有你更精彩!!!
回復

使用道具 舉報

ID:302760 發表于 2018-4-6 18:13 | 顯示全部樓層
學習一下
回復

使用道具 舉報

ID:250045 發表于 2018-4-12 11:54 | 顯示全部樓層
很好的資料,剛好最近在做這個。
回復

使用道具 舉報

ID:311523 發表于 2018-4-19 11:04 | 顯示全部樓層
謝謝分享
回復

使用道具 舉報

ID:324603 發表于 2018-5-7 22:01 | 顯示全部樓層
謝謝分享
回復

使用道具 舉報

ID:326595 發表于 2018-5-10 10:35 | 顯示全部樓層
非常感謝,新手入門,很有用
回復

使用道具 舉報

ID:336452 發表于 2018-5-23 14:24 | 顯示全部樓層
謝謝樓主
回復

使用道具 舉報

ID:197331 發表于 2018-6-6 18:47 | 顯示全部樓層
不錯,值得學習
回復

使用道具 舉報

ID:344257 發表于 2018-6-6 21:09 | 顯示全部樓層
這個源代碼直接放到keil里面編譯,是有警告的啊?
回復

使用道具 舉報

ID:356219 發表于 2018-6-21 17:04 | 顯示全部樓層
謝謝哈哈
回復

使用道具 舉報

ID:358826 發表于 2018-6-25 21:12 | 顯示全部樓層
下來慢慢學
回復

使用道具 舉報

ID:293560 發表于 2018-7-13 10:53 | 顯示全部樓層
pid入坑中
回復

使用道具 舉報

ID:370696 發表于 2018-7-13 11:55 | 顯示全部樓層
謝謝分享
回復

使用道具 舉報

ID:371038 發表于 2018-7-13 17:06 | 顯示全部樓層
非常好
回復

使用道具 舉報

ID:371504 發表于 2018-7-14 15:09 | 顯示全部樓層
值得學習
回復

使用道具 舉報

ID:339697 發表于 2018-8-19 07:49 | 顯示全部樓層
很想試試看
回復

使用道具 舉報

ID:398044 發表于 2018-9-14 17:14 | 顯示全部樓層
很想試試看
回復

使用道具 舉報

ID:402275 發表于 2018-10-10 17:43 | 顯示全部樓層
感謝大佬
回復

使用道具 舉報

ID:405059 發表于 2018-10-13 09:46 | 顯示全部樓層
第一次用PID算法來控制,文件里位置式和增量式都涉及到了,很詳細
回復

使用道具 舉報

ID:432291 發表于 2018-11-24 11:50 | 顯示全部樓層
正好,試著做做
回復

使用道具 舉報

ID:452353 發表于 2018-12-22 21:47 | 顯示全部樓層
謝謝樓主分享
回復

使用道具 舉報

ID:452376 發表于 2018-12-22 22:24 | 顯示全部樓層
努力學習亞
回復

使用道具 舉報

ID:450877 發表于 2018-12-22 22:28 | 顯示全部樓層
好帖子,試試做
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

手機版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 国产精品日韩欧美一区二区三区 | a天堂在线| 犬夜叉在线观看 | 中文字幕动漫成人 | 欧美日韩国产精品激情在线播放 | 一区二区高清 | 本道综合精品 | av在线免费播放 | 九九九久久国产免费 | 久久亚洲二区 | 国产黄色在线 | 久久精品欧美一区二区三区不卡 | 精品国产久 | 成人午夜网站 | 国产午夜精品视频 | 久草中文网 | 精品国产1区2区3区 一区二区手机在线 | 亚洲国产一区二区三区 | 亚洲一区二区三区视频 | 天堂网中文字幕在线观看 | 91麻豆精品国产91久久久更新资源速度超快 | 91精品国产高清一区二区三区 | 国产伦精品一区二区三区高清 | 刘亦菲国产毛片bd | 日韩午夜影院 | 一级一级一级毛片 | 色资源在线视频 | 久久高清| 在线中文字幕第一页 | 一级全黄少妇性色生活免费看 | 日韩激情视频一区 | 亚洲一区二区三区桃乃木香奈 | 日本韩国电影免费观看 | 九九综合| 99久久精品一区二区毛片吞精 | 色综合久久天天综合网 | 欧美白人做受xxxx视频 | 欧美日韩国产一区二区三区 | 欧美精品一区二区三区四区五区 | 日韩伦理一区二区三区 | 91精品国产乱码久久久久久久久 |