找回密码
 立即注册

QQ登录

只需一步,快速开始

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

51单片机水箱检测系统程序 检测温度ph值 PID算法控制温度到设置值

[复制链接]
跳转到指定楼层
楼主
51单片机检测水箱内温度,ph值;
使用pid算法控制温度到设置值;
普通控制ph值到设定值

温度低,启动加热;PH值过高,启动进水;PH值过低,启动进料。
水位过高,启动放水。

电路原理图如下:


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


程序流程图



元件清单:
元件    型号    数量
单片机    AT89C51    1
电容    10uf    1
电容    30pf    2
晶振    12MHZ    1
电阻    10k    1
按钮        7
温度传感器    DS18B20    1
模数芯片    ADC0832    1
电位器    1k    2
电阻    1k    4
三极管    PNP    4
继电器    5V    4
显示器    LCD1602    1
排阻    10k    1

单片机源程序如下:
  1. #include "reg51.h"
  2. #include "lcd1602.h"
  3. #include "Ds18b20.h"
  4. #include "ADC0832.h"
  5. #include"pid_pwm.h"
  6. #define uchar unsigned char
  7. #define uint unsigned int
  8. sbit k1=P3^2;//按钮
  9. sbit k2=P3^3;
  10. sbit k3=P3^4;
  11. sbit k4=P3^5;
  12. sbit k5=P3^6;
  13. sbit k6=P3^7;
  14. sbit out1=P2^4;//输出控制
  15. sbit out2=P2^5;
  16. sbit out3=P2^6;
  17. sbit out4=P2^7;

  18. uchar time=0,mode=0;//系统变量
  19. uchar wendu=0,ph=0,water=0;//温度,pH值,水位
  20. uchar lim1=25,lim2=7,lim3=100;//阀值
  21. uchar disp1[]="T:000 P:00 H:000";
  22. uchar disp2[]="  000   00   000";
  23. uchar pwm=0;
  24. //主函数
  25. void main()
  26. {
  27.         uchar i=0;
  28.         init_1602();
  29.         TMOD|=0X01;
  30.         TH0=0X3C;
  31.         TL0=0XB0;        
  32.         ET0=1;//打开定时器0中断允许
  33.         EA=1;//打开总中断
  34.         TR0=1;//打开定时器
  35.         while(1)
  36.         {
  37.                 //PWM输出
  38.                 if(i<99)
  39.                          i++;
  40.                 else
  41.                         i=0;
  42.                 if(i<pwm)
  43.                         out4=0;
  44.                 else
  45.                         out4=1;
  46.                 //按键检测
  47.                 if(!k1)
  48.                 {
  49.                         if(lim1<120)
  50.                                 lim1++;
  51.                         while(!k1);
  52.                 }
  53.                 if(!k2)
  54.                 {
  55.                         if(lim1>0)
  56.                                 lim1--;
  57.                         while(!k2);
  58.                 }
  59.                 if(!k3)
  60.                 {
  61.                         if(lim2<14)
  62.                                 lim2++;
  63.                         while(!k3);
  64.                 }
  65.                 if(!k4)
  66.                 {
  67.                         if(lim2>0)
  68.                                 lim2--;
  69.                         while(!k4);
  70.                 }
  71.                 if(!k5)
  72.                 {
  73.                         if(lim3<127)
  74.                                 lim3++;
  75.                         while(!k5);
  76.                 }
  77.                 if(!k6)
  78.                 {
  79.                         if(lim3>0)
  80.                                 lim3--;
  81.                         while(!k6);
  82.                 }
  83.         }
  84. }
  85. //定时器中断
  86. void Timer0() interrupt 1
  87. {
  88.         uint i=0;
  89.         if(time<10)//0.5s
  90.                 time++;
  91.         else
  92.         {
  93.                 time=0;
  94.                 Ds18b20ReadTemp();//测温
  95.                 wendu=ds18b20_temp;
  96.                 ph=ADC(1);//PH
  97.                 i=ph;
  98.                 ph=i*14/127;
  99.                 water=ADC(2);//水位
  100.                 //显示
  101.                 disp1[2]=wendu/100+0x30;
  102.                 disp1[3]=wendu%100/10+0x30;
  103.                 disp1[4]=wendu%10+0x30;
  104.                 disp1[8]=ph/10+0x30;
  105.                 disp1[9]=ph%10+0x30;
  106.                 disp1[13]=water/100+0x30;
  107.                 disp1[14]=water%100/10+0x30;
  108.                 disp1[15]=water%10+0x30;
  109.                 write_string(1,0,disp1);
  110.                 disp2[2]=lim1/100+0x30;
  111.                 disp2[3]=lim1%100/10+0x30;
  112.                 disp2[4]=lim1%10+0x30;
  113.                 disp2[8]=lim2/10+0x30;
  114.                 disp2[9]=lim2%10+0x30;
  115.                 disp2[13]=lim3/100+0x30;
  116.                 disp2[14]=lim3%100/10+0x30;
  117.                 disp2[15]=lim3%10+0x30;
  118.                 write_string(2,0,disp2);
  119.                 //PID控制
  120.                 pwm=PID(lim1,wendu);
  121.                 //PH值控制
  122.                 if(ph>lim2)
  123.                         out1=0;
  124.                 else
  125.                         out1=1;
  126.                 if(ph<lim2)
  127.                         out3=0;
  128.                 else
  129.                         out3=1;
  130.                 //水位控制
  131.                 if(water>lim3)
  132.                         out2=0;
  133.                 else
  134.                         out2=1;
  135.         }
  136.         TH0=0X3C;
  137.         TL0=0XB0;
  138. }
复制代码
  1. #include"pid_pwm.h"

  2. uchar P_dat=1,I_dat=1,D_dat=1;//PID参数
  3. uint pid_dat[]={0,0,0,0,0}; //临时数据
  4. //PID计算,返回占空比
  5. uchar PID(uint mub,uint dat)//调整目标+测量数据
  6. {
  7.         uchar i=0;
  8.         uint j=0,sum=0,k=0;
  9.         //存储数据
  10.         for(i=0;i<4;i++)
  11.                 pid_dat[i]=pid_dat[i+1];
  12.         pid_dat[4]=dat;

  13.         //P调节
  14.         j=0;
  15.         if(dat<mub)
  16.         {
  17.                 j=j+(mub-dat)*P_dat;
  18.                 if(j>100)
  19.                         j=100;
  20.         }
  21.         else
  22.         {
  23.                 k=(dat-mub)*P_dat;
  24.                 if(j>k)
  25.                         j=j-k;
  26.         }
  27.         //I调节
  28.         sum=0;
  29.         for(i=0;i<5;i++)
  30.                 sum+=pid_dat[i];
  31.         sum=sum/5;
  32.         if(sum>mub)//减小
  33.         {
  34.                 k=(sum-mub)*I_dat;
  35.                 if(j>k)
  36.                         j=j-k;
  37.         }
  38.         else  //增加
  39.         {
  40.                 k=(mub-sum)*I_dat;
  41.                 j+=k;
  42.                 if(j>100)
  43.                         j=100;
  44.         }
  45.         //D调节
  46.         if(dat>pid_dat[3])//减小
  47.         {
  48.                 k=(dat-pid_dat[3])*D_dat;
  49.                 if(j>k)
  50.                         j=j-k;
  51.         }
  52.         else  //增加
  53.         {
  54.                 k=(pid_dat[3]-dat)*D_dat;
  55.                 j+=k;
  56.                 if(j>100)
  57.                         j=100;
  58.         }
  59.         return j;
  60. }
复制代码

Keil代码与Proteus仿真下载:
51水箱水质温度检测系统.7z (911.22 KB, 下载次数: 102)

评分

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

查看全部评分

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

使用道具 举报

沙发
ID:429944 发表于 2023-12-6 08:15 | 只看该作者
P_dat=1,I_dat=1,D_dat=1;PID参数一直是不变的话是不是有什么不妥?
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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