找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開始

搜索
查看: 4109|回復(fù): 3
收起左側(cè)

單片機(jī)PID控制螺旋槳到特定角度Proteus仿真程序

  [復(fù)制鏈接]
ID:596062 發(fā)表于 2020-12-13 10:32 | 顯示全部樓層 |閱讀模式
我們單片機(jī)設(shè)計(jì)要求我們用51單片機(jī)控制螺旋槳到特定角度,并通過lcd102顯示出來,按鍵設(shè)置角度,電位器通過pcf8591讀取角度
包含pcb文件,程序,仿真,設(shè)計(jì)書等。做的仍有瑕疵希望見諒。

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

單片機(jī)源程序如下:
  1. #include <reg52.h>               
  2. #include "i2c.h"
  3. #include "delay.h"
  4. #include "1602.h"
  5. #include <stdio.h>
  6. #include <math.h>
  7. #define AddWr 0x90   //寫數(shù)據(jù)地址
  8. #define AddRd 0x91   //讀數(shù)據(jù)地址
  9. #define wait_speak 1
  10. #define speaked  0
  11. sbit PWM=P1^2;
  12. sbit led=P1^6;
  13. sbit target_up=P3^0;
  14. sbit target_down=P3^1;
  15. sbit beep=P1^5;       
  16. float target=40.0;
  17. float Kp=0.05,Ki=0.4,Kd=0.15;
  18. unsigned int value,timer1;
  19. char flag=wait_speak;
  20. unsigned char confirtime=0;
  21. extern bit ack;
  22. unsigned char ReadADC(unsigned char Chl);
  23. float pid(float input);
  24. void control();
  25. void Timer1Init();
  26. void Timer0Init();
  27. void change_target();
  28. void  beep_1s();
  29. void display();

  30. bit WriteDAC(unsigned char dat);
  31. void delay100us(int i)   //誤差 0us
  32. {
  33.     unsigned char a,b;
  34.         while(i--)
  35.     for(b=1;b>0;b--)
  36.         for(a=47;a>0;a--);
  37. }
  38. /*------------------------------------------------
  39.               主程序
  40. ------------------------------------------------*/
  41. main()
  42. {


  43. LCD_Init();           //初始化液晶
  44. DelayMs(20);          //延時(shí)有助于穩(wěn)定
  45. LCD_Clear();          //清屏
  46. Timer1Init();
  47.         value=0;
  48.         led=0;
  49. while (1)              //主循環(huán)
  50.   {
  51.   control();       
  52. change_target();

  53.   }
  54. }
  55. /*------------------------------------------------
  56.              讀AD轉(zhuǎn)值程序
  57. 輸入?yún)?shù) Chl 表示需要轉(zhuǎn)換的通道,范圍從0-3
  58. 返回值范圍0-255
  59. ------------------------------------------------*/
  60. unsigned char ReadADC(unsigned char Chl)
  61. {
  62.    unsigned char Val;
  63.    Start_I2c();               //啟動(dòng)總線
  64.    SendByte(AddWr);             //發(fā)送器件地址
  65.      if(ack==0)return(0);
  66.    SendByte(0x40|Chl);            //發(fā)送器件子地址
  67.      if(ack==0)return(0);
  68.    Start_I2c();
  69.    SendByte(AddWr+1);
  70.       if(ack==0)return(0);
  71.    Val=RcvByte();
  72.    NoAck_I2c();                 //發(fā)送非應(yīng)位
  73.    Stop_I2c();                  //結(jié)束總線
  74.   return(Val);
  75. }
  76. float pid(float input)
  77. {
  78.   static float error2,error1,error3,t,Velocity;
  79.   error1 = target-input ;
  80.   Velocity+=(Kp*(error1-error2))+Ki*error1+Kd*(error1-2*error2+error3);
  81.   t=error2;
  82.   error2= error1 ;
  83.   error3 = t;
  84. return Velocity;

  85. }
  86. //電壓分辨率0.0196V
  87. //滿電壓3.9V所以角度分辨率0.0196/3.9*360=1.8度 所以增大角度分辨率要嘛增加滿量程電壓3.9V要嘛減小其對(duì)應(yīng)的角度,其實(shí)就都是增大電壓
  88. void control()
  89. {
  90.   
  91.         unsigned char num=0,i;
  92.         int sum_num=0;
  93.   unsigned char temp[7];//定義顯示區(qū)域臨時(shí)存儲(chǔ)數(shù)組
  94.         float Voltage,angle;        //定義浮點(diǎn)變量
  95.         float temp_value;
  96.         for(i=0;i<5;i++)
  97.   {   
  98.                 sum_num+=ReadADC(0);   
  99.         }       
  100.         num=sum_num/5;
  101.         sum_num=0;
  102.   Voltage=(float)num;   
  103.   angle=(Voltage-16);//*1.41-23.91);
  104.   //angle=-(((Voltage-4.96)/0.0136)+341.73);
  105.         //pid計(jì)算
  106.         temp_value=(int)pid(angle);
  107.                 if(temp_value<0)
  108.                 value=0;
  109.         else if(temp_value>=100)
  110.                 value=100;
  111.         else
  112.         value=(int)temp_value;

  113.        
  114.         if(flag==wait_speak&&abs(angle-target)<=5)
  115.                    confirtime++;
  116.   if(abs(angle-target)>=20)  //如果角度差大于20度 將嗡鳴器設(shè)置為待激活狀態(tài)
  117.                 flag=wait_speak;
  118.         if(confirtime==10&&(flag==wait_speak))
  119.         {       
  120.                 //beep_1s();
  121.                 sprintf(temp,"Achieve the goal:");//格式輸出電壓值,%3.2f 表示浮點(diǎn)輸出,共3位數(shù),小數(shù)點(diǎn)后2位  
  122.           LCD_Write_String(0,1,temp);
  123.                 delay100us(10000);
  124.                  LCD_Clear();          //清屏
  125.                 flag=speaked;
  126.                 confirtime=0;
  127.         }
  128.        
  129.        

  130.        
  131.   sprintf(temp,"DQJD:%3.2f ", angle);
  132.   LCD_Write_String(0,0,temp);
  133.   sprintf(temp,"SDJD:%d",(int)target);
  134.         LCD_Write_String(0,1,temp);
  135.         //sprintf(temp,"PWM:%d", value);
  136.         //LCD_Write_String(10,1,temp);
  137. }
  138. //定時(shí)器1服務(wù)函數(shù)
  139. void Time1(void) interrupt 3   
  140. {
  141.         TH1 = 0x0FF;
  142.         TL1 = 0x9C;
  143.         timer1++;
  144.                 if(timer1>100)  //PWM周期為100*100us
  145.                 {
  146.                         timer1=0;
  147.                 }
  148.                 if(timer1 <value)       
  149.                 {
  150.                         PWM=1;
  151.                 }
  152.                 else
  153.                 {
  154.                         PWM=0;
  155.                 }
  156.        
  157.        
  158. }
  159. //定時(shí)器1初始化函數(shù)
  160. void Timer1Init()
  161. {
  162.         TMOD|=0X10;//選擇為定時(shí)器1模式,工作方式1,僅用TR1打開啟動(dòng)。
  163.         TH1 = 0x0FF;
  164.   TL1 = 0x9C;       
  165.         ET1=1;//打開定時(shí)器1中斷允許
  166.         EA=1;//打開總中斷
  167.         TR1=1;//打開定時(shí)器                       
  168. }

  169. void change_target()
  170. {
  171. unsigned char temp[7];//定義顯示區(qū)域臨時(shí)存儲(chǔ)數(shù)組
  172.         if(target_up==0)
  173.         {
  174.                 delay100us(100);
  175.                 if(target_up==0)
  176.                 {         flag=wait_speak;
  177.                         while(target_up==0)
  178.                         {               
  179.         if(target<=80)               
  180.                                 target=45;               
  181.                     sprintf(temp,"Target:%d",(int)target);//格式輸出電壓值,%3.2f 表示浮點(diǎn)輸出,共3位數(shù),小數(shù)點(diǎn)后2位  
  182.               LCD_Write_String(0,1,temp);
  183.                           delay100us(1000);
  184.                        
  185.                         }
  186.                 }
  187.         }
  188.         if(target_down==0)
  189.         {
  190.                 delay100us(100);
  191.                 if(target_down==0)
  192.                 {        flag=wait_speak;
  193.                         while(target_down==0)
  194.                         {                               
  195.                                  if(target>=10)
  196.                                 target=0;               
  197.                     sprintf(temp,"Target:%d",(int)target);//格式輸出電壓值,%3.2f 表示浮點(diǎn)輸出,共3位數(shù),小數(shù)點(diǎn)后2位  
  198.               LCD_Write_String(0,1,temp);
  199.                           //delay10us(1000);
  200.                        
  201.                         }
  202.                 }
  203.         }
  204. }
  205. void display()
  206. {
  207. //        unsigned char temp[7];//定義顯示區(qū)域臨時(shí)存儲(chǔ)數(shù)組
  208. //  sprintf(temp,"Reality: %3.2f ", angle);//格式輸出電壓值,%3.2f 表示浮點(diǎn)輸出,共3位數(shù),小數(shù)點(diǎn)后2位
  209. //  LCD_Write_String(0,0,temp);
  210. //  sprintf(temp,"Target:%d",(int)target);//格式輸出電壓值,%3.2f 表示浮點(diǎn)輸出,共3位數(shù),小數(shù)點(diǎn)后2位  
  211. //        LCD_Write_String(0,1,temp);
  212. //        sprintf(temp,"PWM:%d", value);//格式輸出電壓值,%3.2f 表示浮點(diǎn)輸出,共3位數(shù),小數(shù)點(diǎn)后2位  
  213. //        LCD_Write_String(9,1,temp);

  214. }
  215. void  beep_1s()
  216. {
  217.         unsigned char i;
  218. for(i=0;i<=200;i++)
  219.   {
  220.         delay100us(1);
  221.         beep=~beep;
  222.         }

  223. }
復(fù)制代碼

所有資料51hei提供下載:
新建 好壓 ZIP 壓縮文件.zip (1.38 MB, 下載次數(shù): 114)

評(píng)分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎(jiǎng)勵(lì)!

查看全部評(píng)分

回復(fù)

使用道具 舉報(bào)

ID:230742 發(fā)表于 2020-12-13 19:30 | 顯示全部樓層
我是小白,看過很多設(shè)計(jì)。看了你的設(shè)計(jì),我沒明白的是用讀取電位器角度。沒搞錯(cuò)吧。你設(shè)計(jì)的是一個(gè)飛機(jī)啊。難道你要在飛機(jī)的電位器上掛個(gè)重錘?來讀取電位器的角度?
回復(fù)

使用道具 舉報(bào)

ID:47286 發(fā)表于 2021-11-16 10:00 來自手機(jī) | 顯示全部樓層
啤酒瓶子老大 發(fā)表于 2020-12-13 19:30
我是小白,看過很多設(shè)計(jì)。看了你的設(shè)計(jì),我沒明白的是用讀取電位器角度。沒搞錯(cuò)吧。你設(shè)計(jì)的是一個(gè)飛機(jī)啊。 ...

我想重錘是不用 但用電位器的話 轉(zhuǎn)速得很低吧 轉(zhuǎn)速高了采集不對(duì)
回復(fù)

使用道具 舉報(bào)

ID:658144 發(fā)表于 2022-6-23 11:11 | 顯示全部樓層
支持51黑,感謝分享!
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

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

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表