找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机PID控制螺旋桨到特定角度Proteus仿真程序

  [复制链接]
跳转到指定楼层
楼主
我们单片机设计要求我们用51单片机控制螺旋桨到特定角度,并通过lcd102显示出来,按键设置角度,电位器通过pcf8591读取角度
包含pcb文件,程序,仿真,设计书等。做的仍有瑕疵希望见谅。

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


单片机源程序如下:
  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   //写数据地址
  8. #define AddRd 0x91   //读数据地址
  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);          //延时有助于稳定
  45. LCD_Clear();          //清屏
  46. Timer1Init();
  47.         value=0;
  48.         led=0;
  49. while (1)              //主循环
  50.   {
  51.   control();       
  52. change_target();

  53.   }
  54. }
  55. /*------------------------------------------------
  56.              读AD转值程序
  57. 输入参数 Chl 表示需要转换的通道,范围从0-3
  58. 返回值范围0-255
  59. ------------------------------------------------*/
  60. unsigned char ReadADC(unsigned char Chl)
  61. {
  62.    unsigned char Val;
  63.    Start_I2c();               //启动总线
  64.    SendByte(AddWr);             //发送器件地址
  65.      if(ack==0)return(0);
  66.    SendByte(0x40|Chl);            //发送器件子地址
  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();                 //发送非应位
  73.    Stop_I2c();                  //结束总线
  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要嘛减小其对应的角度,其实就都是增大电压
  88. void control()
  89. {
  90.   
  91.         unsigned char num=0,i;
  92.         int sum_num=0;
  93.   unsigned char temp[7];//定义显示区域临时存储数组
  94.         float Voltage,angle;        //定义浮点变量
  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计算
  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度 将嗡鸣器设置为待激活状态
  117.                 flag=wait_speak;
  118.         if(confirtime==10&&(flag==wait_speak))
  119.         {       
  120.                 //beep_1s();
  121.                 sprintf(temp,"Achieve the goal:");//格式输出电压值,%3.2f 表示浮点输出,共3位数,小数点后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. //定时器1服务函数
  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. //定时器1初始化函数
  160. void Timer1Init()
  161. {
  162.         TMOD|=0X10;//选择为定时器1模式,工作方式1,仅用TR1打开启动。
  163.         TH1 = 0x0FF;
  164.   TL1 = 0x9C;       
  165.         ET1=1;//打开定时器1中断允许
  166.         EA=1;//打开总中断
  167.         TR1=1;//打开定时器                       
  168. }

  169. void change_target()
  170. {
  171. unsigned char temp[7];//定义显示区域临时存储数组
  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 表示浮点输出,共3位数,小数点后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 表示浮点输出,共3位数,小数点后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];//定义显示区域临时存储数组
  208. //  sprintf(temp,"Reality: %3.2f ", angle);//格式输出电压值,%3.2f 表示浮点输出,共3位数,小数点后2位
  209. //  LCD_Write_String(0,0,temp);
  210. //  sprintf(temp,"Target:%d",(int)target);//格式输出电压值,%3.2f 表示浮点输出,共3位数,小数点后2位  
  211. //        LCD_Write_String(0,1,temp);
  212. //        sprintf(temp,"PWM:%d", value);//格式输出电压值,%3.2f 表示浮点输出,共3位数,小数点后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. }
复制代码

所有资料51hei提供下载:
新建 好压 ZIP 压缩文件.zip (1.38 MB, 下载次数: 110)

评分

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

查看全部评分

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

使用道具 举报

沙发
ID:230742 发表于 2020-12-13 19:30 | 只看该作者
我是小白,看过很多设计。看了你的设计,我没明白的是用读取电位器角度。没搞错吧。你设计的是一个飞机啊。难道你要在飞机的电位器上挂个重锤?来读取电位器的角度?
回复

使用道具 举报

板凳
ID:47286 发表于 2021-11-16 10:00 来自手机 | 只看该作者
啤酒瓶子老大 发表于 2020-12-13 19:30
我是小白,看过很多设计。看了你的设计,我没明白的是用读取电位器角度。没搞错吧。你设计的是一个飞机啊。 ...

我想重锤是不用 但用电位器的话 转速得很低吧 转速高了采集不对
回复

使用道具 举报

地板
ID:658144 发表于 2022-6-23 11:11 | 只看该作者
支持51黑,感谢分享!
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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