找回密码
 注册

QQ登录

只需一步,快速开始

搜索

51单片机鱼缸温度控制器(程序以及protues仿真在压缩包里)

[复制链接]
coolice 发表于 2020-1-21 17:43:04 | 显示全部楼层 |阅读模式
随着人们生活水平的不断提高,目前各式各样的观赏鱼缸之类的工艺产品逐渐进入了家庭和宾馆、商场等公共场所,由于现有的观赏鱼缸的水温检测、水温控制等操作都特别繁琐,给人们带来了很大的不便。本文通过对目前大多数鱼缸控制设备应用现状的分析和研究,提出了一种鱼缸温度智能检测系统的设计方案。针对目前大多数鱼缸控制设备价格昂贵、安装繁琐、运行费用高,一般的用户难以使用的情况,我们结合单片机强大的开发技术,设计制作了一种以单片机为控制核心传感器技术进行温度显示并实现智能温度检测。此系统主要以STC89C51单片机检测系统为核心实现对鱼缸的集中控制和管理,能够对鱼缸温度进行自动检测,温度实时显示用LCD显示屏实现,用DS18B20温度传感器完成实时传送温度的指标。

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
智能鱼缸.zip (113.09 KB, 售价: 3 E币)

1.jpg
  1. #include <reg51.h>
  2. #define uint unsigned int
  3. #define uchar unsigned char           //宏定义
  4. #define LCD1602 P0
  5. sbit SET=P3^1;                            //定义调整键
  6. sbit DEC=P3^2;                            //定义减少键
  7. sbit ADD=P3^3;                            //定义增加键
  8. sbit BUZZ=P3^6;                            //定义蜂鸣器
  9. sbit ALAM=P1^2;                                //定义灯光报警
  10. sbit ALAM1=P1^4;
  11. sbit DQ=P3^7;                             //定义DS18B20总线I/O        
  12. sbit RS = P2^7;
  13. sbit EN = P2^6;
  14. bit shanshuo_st;                            //闪烁间隔标志
  15. bit beep_st;                                     //蜂鸣器间隔标志
  16. uchar x=0;                                      //计数器

  17. uchar code tab1[]={"Now Tem:   .  C "};
  18. uchar code tab2[]={"TH:   C  TL:   C"};
  19. uint c;
  20. uchar Mode=0;                             //状态标志
  21. signed char TH=40;                  //上限报警温度,默认值为40
  22. signed char TL=10;                   //下限报警温度,默认值为10
  23. //============================================================================================
  24. //====================================DS18B20=================================================
  25. //============================================================================================
  26. /*****延时子程序*****/
  27. void Delay_DS18B20(int num)
  28. {
  29.   while(num--) ;
  30. }
  31. void delay(uint xms)//延时函数,有参函数
  32. {
  33.         uint x,y;
  34.         for(x=xms;x>0;x--)
  35.          for(y=110;y>0;y--);
  36. }
  37. /*****初始化DS18B20*****/
  38. void Init_DS18B20(void)
  39. {
  40.   unsigned char x=0;
  41.   DQ = 1;         //DQ复位
  42.   Delay_DS18B20(8);    //稍做延时
  43.   DQ = 0;         //单片机将DQ拉低
  44.   Delay_DS18B20(80);   //精确延时,大于480us
  45.   DQ = 1;         //拉高总线
  46.   Delay_DS18B20(14);
  47.   x = DQ;           //稍做延时后,如果x=0则初始化成功,x=1则初始化失败
  48.   Delay_DS18B20(20);
  49. }
  50. /*****读一个字节*****/
  51. unsigned char ReadOneChar(void)
  52. {
  53.   unsigned char i=0;
  54.   unsigned char dat = 0;
  55.   for (i=8;i>0;i--)
  56.   {
  57.     DQ = 0;     // 给脉冲信号
  58.     dat>>=1;
  59.     DQ = 1;     // 给脉冲信号
  60.     if(DQ)
  61.     dat|=0x80;
  62.     Delay_DS18B20(4);
  63.   }
  64.   return(dat);
  65. }
  66. /*****写一个字节*****/
  67. void WriteOneChar(unsigned char dat)
  68. {
  69.   unsigned char i=0;
  70.   for (i=8; i>0; i--)
  71.   {
  72.     DQ = 0;
  73.     DQ = dat&0x01;
  74.     Delay_DS18B20(5);
  75.     DQ = 1;
  76.     dat>>=1;
  77.   }
  78. }
  79. /*****读取温度*****/
  80. unsigned int ReadTemperature(void)
  81. {
  82.   unsigned char a=0;
  83.   unsigned char b=0;
  84.   unsigned int t=0;
  85.   float tt=0;
  86.   Init_DS18B20();
  87.   WriteOneChar(0xCC);  //跳过读序号列号的操作
  88.   WriteOneChar(0x44);  //启动温度转换
  89.   Init_DS18B20();
  90.   WriteOneChar(0xCC);  //跳过读序号列号的操作
  91.   WriteOneChar(0xBE);  //读取温度寄存器
  92.   a=ReadOneChar();     //读低8位
  93.   b=ReadOneChar();    //读高8位
  94.   t=b;
  95.   t<<=8;
  96.   t=t|a;
  97.   tt=t*0.0625;
  98. // t= tt*10+0.5;     //放大10倍输出并四舍五入
  99.   t= tt*10+0.5;
  100.   return(t);
  101. }

  102. /*****读取温度*****/
  103. void check_wendu(void)
  104. {
  105.         c=ReadTemperature()-5;                          //获取温度值并减去DS18B20的温漂误差
  106.         if(c>1200)
  107.         c=1200;
  108. }

  109. /********液晶写入指令函数与写入数据函数,以后可调用**************/

  110. void write_1602com(uchar com)//****液晶写入指令函数****
  111. {
  112.         RS=0;//数据/指令选择置为指令
  113. //        rw=0; //读写选择置为写
  114.         LCD1602=com;//送入数据
  115.         delay(1);
  116.         EN=1;//拉高使能端,为制造有效的下降沿做准备
  117.         delay(1);
  118.         EN=0;//en由高变低,产生下降沿,液晶执行命令
  119. }


  120. void write_1602dat(uchar dat)//***液晶写入数据函数****
  121. {
  122.         RS=1;//数据/指令选择置为数据
  123. //        rw=0; //读写选择置为写
  124.         LCD1602=dat;//送入数据
  125.         delay(1);
  126.         EN=1; //en置高电平,为制造下降沿做准备
  127.         delay(1);
  128.         EN=0; //en由高变低,产生下降沿,液晶执行命令
  129. }


  130. void lcd_init()//***液晶初始化函数****
  131. {
  132.         uchar a;
  133.         write_1602com(0x38);//设置液晶工作模式,意思:16*2行显示,5*7点阵,8位数据
  134.         write_1602com(0x0c);//开显示不显示光标
  135.         write_1602com(0x06);//整屏不移动,光标自动右移
  136.         write_1602com(0x01);//清显示

  137.         write_1602com(0x80);//日历显示固定符号从第一行第1个位置之后开始显示
  138.         for(a=0;a<16;a++)
  139.         {
  140.                 write_1602dat(tab1[a]);//向液晶屏写日历显示的固定符号部分
  141.                 delay(3);
  142.         }
  143.         write_1602com(0x80+0x40);//时间显示固定符号写入位置,从第2个位置后开始显示
  144.         for(a=0;a<16;a++)
  145.         {
  146.                 write_1602dat(tab2[a]);//写显示时间固定符号,两个冒号
  147.                 delay(3);
  148.         }

  149. }

  150. void display()
  151. {
  152.         if(Mode==0)
  153.         {
  154.                 write_1602com(0x80+8);
  155.                 write_1602dat(c/1000+0x30);
  156.                 write_1602dat((c%1000)/100+0x30);
  157.                 write_1602dat(((c%1000)%100)/10+0x30);
  158.                 write_1602com(0x80+12);
  159.                 write_1602dat(((c%1000)%100)%10+0x30);
  160.                 write_1602com(0x80+13);
  161.                 write_1602dat(0xdf);
  162.                 write_1602com(0x80+0x40+3);
  163.                 write_1602dat(TH/10+0x30);
  164.                 write_1602dat(TH%10+0x30);
  165.                 write_1602dat(0xdf);
  166.                 write_1602com(0x80+0x40+12);
  167.                 write_1602dat(TL/10+0x30);
  168.                 write_1602dat(TL%10+0x30);
  169.                 write_1602dat(0xdf);                        
  170.         }                                                                  
  171. }
  172. //=====================================================================================

  173. /*****初始化定时器0*****/
  174. void InitTimer(void)
  175. {
  176.         TMOD=0x1;
  177.         TH0=0x3c;
  178.         TL0=0xb0;     //50ms(晶振12M)
  179.         EA=1;      //全局中断开关
  180.         TR0=1;
  181.         ET0=1;      //开启定时器0
  182. }

  183. void KEY()
  184. {
  185.                         //功能键
  186.         if(SET==0)
  187.         {
  188.                 BUZZ=0;
  189.                 delay(10);
  190.                 if(SET==0)
  191.                 {
  192.                         Mode++;
  193.                         if(Mode==3)
  194.                         Mode=0;
  195.                         BUZZ=1;
  196.                 }
  197.                 while(SET==0)
  198.                 {
  199.                         if(Mode==0)
  200.                                 {
  201.                                 //        write_1602com(0x80+0x40+6);
  202.                                         write_1602com(0x0c);
  203.                                 }        
  204.                         else if(Mode==1)
  205.                                 {
  206.                                         write_1602com(0x80+0x40+4);
  207.                                         write_1602com(0x0f);
  208.                                 }        
  209.                         else
  210.                                 {
  211.                                         write_1602com(0x80+0x40+13);
  212.                                         write_1602com(0x0f);
  213.                                 }                                                        
  214.                 }
  215.         }
  216.         //增加
  217.         if(ADD==0&&Mode==1)
  218.         {
  219.                 BUZZ=0;
  220.                 delay(10);
  221.                 if(ADD==0)        
  222.                 {
  223.                         TH++;
  224.                         if(TH>=99)        
  225.                         TH=99;
  226.                         write_1602com(0x80+0x40+3);
  227.                         write_1602dat(TH/10+0x30);
  228.                         write_1602dat(TH%10+0x30);
  229.                         write_1602com(0x80+0x40+4);        
  230.                         BUZZ=1;
  231.                 }
  232.                 while(ADD==0);
  233.                
  234.         }
  235.         //减少
  236.         if(DEC==0&&Mode==1)
  237.         {
  238.                 BUZZ=0;
  239.                 delay(10);
  240.                 if(DEC==0)
  241.                 {
  242.                         TH--;
  243.                         if(TH==TL)        
  244.                         TH=TL+1;
  245.                         write_1602com(0x80+0x40+3);
  246.                         write_1602dat(TH/10+0x30);
  247.                         write_1602dat(TH%10+0x30);
  248.                         write_1602com(0x80+0x40+4);        
  249.                         BUZZ=1;
  250.                 }
  251.                 while(DEC==0);
  252.         }
  253.         if(ADD==0&&Mode==2)
  254.         {
  255.                 BUZZ=0;
  256.                 delay(10);
  257.                 if(ADD==0)        
  258.                 {
  259.                         TL++;
  260.                         if(TL==TH)        
  261.                         TL=TH-1;
  262.                         write_1602com(0x80+0x40+12);
  263.                         write_1602dat(TL/10+0x30);
  264.                         write_1602dat(TL%10+0x30);
  265.                         write_1602com(0x80+0x40+13);        
  266.                         BUZZ=1;
  267.                 }
  268.                 while(ADD==0);
  269.                
  270.         }
  271.         //减少
  272.         if(DEC==0&&Mode==2)
  273.         {
  274.                 BUZZ=0;
  275.                 delay(10);
  276.                 if(DEC==0)
  277.                 {
  278.                         TL--;
  279.                         if(TL<=0)        
  280.                         TL=0;
  281.                         write_1602com(0x80+0x40+12);
  282.                         write_1602dat(TL/10+0x30);
  283.                         write_1602dat(TL%10+0x30);
  284.                         write_1602com(0x80+0x40+13);        
  285.                         BUZZ=1;
  286.                 }
  287.                 while(DEC==0);               
  288.         }
  289. }

  290. /*****报警子程序*****/
  291. void Alarm()
  292. {
  293.         if(x>=10){beep_st=~beep_st;x=0;}
  294.         if(Mode==0)
  295.         {
  296.                 if((c/10)>=TH)
  297.                 {
  298.                         ALAM=0;
  299.                         ALAM1=1;
  300.                         if(beep_st==1)
  301.                         BUZZ=0;
  302.                         else
  303.                         BUZZ=1;
  304.                 }
  305.                 else if((c/10)<TL)
  306.                 {
  307.                         ALAM1=0;
  308.                         ALAM=1;
  309.                         if(beep_st==1)
  310.                         BUZZ=0;
  311.                         else
  312.                         BUZZ=1;
  313.                 }
  314.                 else
  315.                 {
  316.                         BUZZ=1;
  317.                         ALAM=1;
  318.                         ALAM1=1;               
  319.                 }
  320.         }
  321.         else
  322.         {
  323.                 BUZZ=1;
  324.                 ALAM=1;
  325.                 ALAM1=1;
  326.         }
  327. }

  328. /*****主函数*****/
  329. void main(void)
  330. {
  331.         uint z;
  332.         delay(1);
  333.         lcd_init();
  334.         delay(1);
  335.         InitTimer();    //初始化定时器
  336.         
  337.         for(z=0;z<100;z++)
  338.         {
  339.                 check_wendu();
  340.                 delay(1);        
  341.         }
  342.         while(1)
  343.         {
  344.                
  345.                 display();
  346.                 KEY();
  347.                 Alarm();
  348.                 check_wendu();
  349.         }
  350. }

  351. /*****定时器0中断服务程序*****/
  352. void timer0(void) interrupt 1
  353. {
  354. TH0=0x3c;
  355. TL0=0xb0;
  356. x++;
  357. }
复制代码





一听柚子 发表于 2022-6-13 17:26:43 | 显示全部楼层
请问大佬为什么我的数码管显示一直是120,有解决方法吗?
 楼主| coolice 发表于 2022-6-14 01:18:28 | 显示全部楼层
一听柚子 发表于 2022-6-13 17:26
请问大佬为什么我的数码管显示一直是120,有解决方法吗?

DS18B20的数值是仿真的,可以在仿真电路这个原件上点击那上下“箭头”进行数值调整。
1.jpg
一听柚子 发表于 2022-6-17 09:08:05 | 显示全部楼层
coolice 发表于 2022-6-14 01:18
DS18B20的数值是仿真的,可以在仿真电路这个原件上点击那上下“箭头”进行数值调整。

...

这个我是知道的,上下箭头点击之后无效。一直显示120,不知道怎么解决。
258B`V([HN4VI`ASWWEV3CE.png
 楼主| coolice 发表于 2022-6-18 01:17:07 | 显示全部楼层
一听柚子 发表于 2022-6-17 09:08
这个我是知道的,上下箭头点击之后无效。一直显示120,不知道怎么解决。 ...

这个问题从来没遇到过,实在不行,把当前的protues卸载干净,换个版本试试。我安装的版本没问题。

1.png
您需要登录后才可以回帖 登录 | 注册

本版积分规则

QQ|手机版|小黑屋|ELEOK |网站地图

GMT+8, 2025-1-21 04:45

Powered by Discuz! X3.5

© 2001-2025 Discuz! Team.

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