找回密码
 立即注册

QQ登录

只需一步,快速开始

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

基于51单片机的温度控制系统Proteus仿真+代码

  [复制链接]
跳转到指定楼层
#
ID:269960 发表于 2019-1-14 11:47 | 只看该作者 回帖奖励 |正序浏览 |阅读模式
摘 要:单片机技术已经普及到我们生活,工作,科研,各个领域,已经成为一种比较成熟的技术,本文将介绍一种基于单片机保温箱的温控系统研究,本恒温箱属于多功能多用途,可以设置上下控制温度,当温度不在设置范围内时,可以驱动相应的负载工作,同时声光报警。

随着现代工农业技术的发展及人们对生活环境要求的提高,人们也迫切需要检测与控制温度。本文通过采用继电器作为温度的主要控制元件,它可以直接驱动2500W功率的负载,可以应用于家庭、小型工厂等小电量用电设备,亦可以用继电器来控制交流接触器线圈等,就可以实现对大功率负载的控制,应用十分广泛。采用蜂鸣器作为电声元件的报警,LED发光指示相应的加热或制冷工作。这种保温箱的温控系统结构简单,可操作性强,应用广泛。工作时,温度控制范围为上下限之间,当前环境温度若超过设定的临界温度,由单片机发出控制和报警信号,从而负载控制温度的变化。
基于单片机保温箱的温控系统大部分使用是在实验室、工业、医药、农业中。在实验室中特别是生物实验室,我们为了得到更加准确的实验数据,对于恒温实验环境要求严格。所以针对实验室来说,恒温箱的作用显得相当重要。在工业生产过程中我们对于恒温箱的要求也相对更加严格,比如产品的热处理、冷处理等,直接影响着产品的质量。在医药方面医用恒温箱主要用于药品和试剂的储存、运输,疫苗、血液的冷藏保温,透析液的加温、生理盐水的加温等。 由以上我们可以明显的看出恒温箱的重要作用。在农业温室大棚中,温控系统对于农作物的生长至关重要,对于农业方面,以至于生活中的各个方面温控系统永远处于相当重要的地位。

主要实现:实时温度测量及显示,超出温度范围相应的继电器工作,继电器可以驱动相应的加热或制冷负载,上下限温度可通过按键设定等功能。
本保温箱的温控系统研究是基于51单片机及温度传感器DS18B20来设计的,温度测量范围0到99.9摄氏度,精度为0.1摄氏度,可见测量温度的范围广,精度高的特点。可设置上下限温度,默认上限温度为38℃、默认下限温度为5℃(通过程序可以更改上下限初始值)。报警值可设置范围:最低上限报警值等于当前下限报警值,最高下限报警值等于当前上限报警值。将下限报警值调为0时为关闭下限报警功能。开启相应的继电器工作时,有指示灯可以指示相应的加热和制冷。


单片机源程序如下:
  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. }
复制代码

所有资料51hei提供下载:
基于单片机的温控系统.zip (111.73 KB, 下载次数: 1958)


评分

参与人数 2黑币 +60 收起 理由
zbfdyw + 10 资料非常好!
admin + 50 共享资料的黑币奖励!

查看全部评分

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

使用道具 举报

来自 116#
ID:328014 发表于 2020-12-28 00:43 | 只看该作者
zzzxzx 发表于 2020-12-28 00:16
我的Led 不工作啊

原理图里面并没有led啊,只有一个LCD,我刚测试了下,楼主的程序和电路是没有问题的,仿真成功了,如图,我用的的Proteus7.8版本的,我估计是你版本不是7.8的问题

51hei.png (18.01 KB, 下载次数: 842)

51hei.png
回复

使用道具 举报

来自 115#
ID:496636 发表于 2022-6-25 13:26 | 只看该作者
我来说一下工业生产中的情况吧,如果想温度控制精度在+-0.1度,那你的温度计必须能够分辩0.01度,DS18B20这个温度计不够精确的,如果只是单片机学习学习,那还行,还有在粗放的温度控制中,还能应用, 一般在工业生产中,不会用这温度计。我们厂里的老师傅对这货理都不理,太垃圾的东西。一般都是用PT100热电阻,用了一款放大器,具体什么集成块,有空找块板子看一下。十块不到的价格,还有好几路。调较过后,精度和稳定性很好,最多在0.01上下跳动。

评分

参与人数 1黑币 +40 收起 理由
admin + 40 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

114#
ID:1144243 发表于 2025-5-26 14:40 来自触屏版 | 只看该作者
厉害有用,太厉害了
回复

使用道具 举报

113#
ID:1151673 发表于 2025-5-26 09:49 | 只看该作者
资料很不错
回复

使用道具 举报

112#
ID:1068711 发表于 2023-6-13 11:28 | 只看该作者
天涯此明明 发表于 2023-3-17 16:31
程序一上电,就会报警一次,降温电路的灯会亮一下。,所以有BUG,但是看了程序不晓得在哪里改,我只有改了 ...

您好,请问红灯怎么亮啊
回复

使用道具 举报

111#
ID:1068711 发表于 2023-6-13 11:00 | 只看该作者
楼主您好,我在仿真的时候温度传感器的值超出上下限,但是红色LED不工作,蜂鸣器也一直响,请问是哪里出错了,求指教~
回复

使用道具 举报

110#
ID:1073424 发表于 2023-4-22 18:47 | 只看该作者
非常好,很有借鉴意义
回复

使用道具 举报

109#
ID:1035505 发表于 2023-4-19 17:02 来自触屏版 | 只看该作者
我的仿真还是不亮也不响,程序显示没错误,实在搞不来了
回复

使用道具 举报

108#
ID:1067664 发表于 2023-3-22 17:29 | 只看该作者
谢问楼主,这个0.5的温度差怎么消除啊
回复

使用道具 举报

107#
ID:1064641 发表于 2023-3-17 16:31 | 只看该作者
程序一上电,就会报警一次,降温电路的灯会亮一下。,所以有BUG,但是看了程序不晓得在哪里改,我只有改了硬件的继电器驱动了,可以驱动两个继电器了
回复

使用道具 举报

106#
ID:1059243 发表于 2022-12-26 16:28 | 只看该作者
这个真的很不错
回复

使用道具 举报

105#
ID:1035505 发表于 2022-7-5 16:18 来自触屏版 | 只看该作者
完全按照程序来的并生成了HEX文件,但是没有发出报警声,LED-YELLOW也不亮是为什么?
回复

使用道具 举报

104#
ID:1035505 发表于 2022-7-3 17:17 来自触屏版 | 只看该作者
这个标号为Q1的图名字是什么?
回复

使用道具 举报

103#
ID:1035250 发表于 2022-6-20 17:18 来自触屏版 | 只看该作者
为什么没有报警,继电器都没工作
回复

使用道具 举报

102#
ID:1035250 发表于 2022-6-16 22:19 来自触屏版 | 只看该作者
我想问你个问题,为什么仿真的时候继电器都不工作,二极管都不亮
回复

使用道具 举报

101#
ID:868264 发表于 2022-4-15 15:11 | 只看该作者
brew9 发表于 2019-4-18 17:35
很想知道如何以0.5度误差的DS18B20,来实现0.1度的温控精度。

为什么他的仿真显示误差这么大?
回复

使用道具 举报

100#
ID:866517 发表于 2021-11-29 22:53 | 只看该作者
感谢楼主,正好有这部分需要
回复

使用道具 举报

99#
ID:914633 发表于 2021-7-6 18:14 | 只看该作者
wmwlw 发表于 2020-1-14 16:14
这个测量温度精度怎么看啊?

ds18b20的配置寄存器可以设置精度,有四个选择0.5、0.25、0.125、0.0625,默认为0.0625
回复

使用道具 举报

98#
ID:914633 发表于 2021-7-6 18:12 | 只看该作者
brew9 发表于 2019-4-18 17:35
很想知道如何以0.5度误差的DS18B20,来实现0.1度的温控精度。

0.5只是它的误差,ds18b20最高精度可达到0.0625
回复

使用道具 举报

97#
ID:947222 发表于 2021-7-6 14:13 | 只看该作者
最近正好要用到这个程序,楼主帮了大忙,谢谢了
回复

使用道具 举报

96#
ID:940811 发表于 2021-6-20 18:19 来自触屏版 | 只看该作者
博主写的很好,但是最好说一下各部分的功能
回复

使用道具 举报

95#
ID:924005 发表于 2021-6-18 20:27 来自触屏版 | 只看该作者
RoseCG 发表于 2019-4-22 09:02
怎么仿真的led没有显示啊

我这也是没有反应
回复

使用道具 举报

94#
ID:922168 发表于 2021-6-15 11:10 | 只看该作者
他这个为什么打开后会一直在报警
回复

使用道具 举报

93#
ID:923340 发表于 2021-5-18 16:03 | 只看该作者
资料非常好用,学习到了
回复

使用道具 举报

92#
ID:897014 发表于 2021-3-27 14:46 | 只看该作者
我想做个实物,楼主能不能发一个元器件清单,我不知道该买哪些元器件
回复

使用道具 举报

91#
ID:896500 发表于 2021-3-25 21:25 来自触屏版 | 只看该作者
rainApril 发表于 2021-3-18 14:39
感谢,刚刚用到了

麻烦问一下LCD不显示是怎么回事
回复

使用道具 举报

90#
ID:893241 发表于 2021-3-18 14:39 | 只看该作者
感谢,刚刚用到了
回复

使用道具 举报

89#
ID:848824 发表于 2021-1-27 17:51 来自触屏版 | 只看该作者
不错,很好我也仿制好了。
回复

使用道具 举报

88#
ID:848824 发表于 2021-1-16 12:50 | 只看该作者
太感谢了,高手很成功做了一个,请问怎么能设置零下50度呢
回复

使用道具 举报

87#
ID:848824 发表于 2021-1-13 12:05 来自触屏版 | 只看该作者
现在网上卖的温控器8块钱的,是什么单片机啊!电路很简单
回复

使用道具 举报

86#
ID:848824 发表于 2021-1-12 08:12 来自触屏版 | 只看该作者
请问一下怎么能设置零下50度°
回复

使用道具 举报

85#
ID:848824 发表于 2021-1-10 22:33 | 只看该作者
怎么能调零下50度啊
回复

使用道具 举报

84#
ID:848824 发表于 2021-1-10 18:54 来自触屏版 | 只看该作者
太好了,感谢大神。一次成功
回复

使用道具 举报

83#
ID:848824 发表于 2021-1-9 23:54 | 只看该作者
x下载看看怎么样,刚买的屏幕试试
回复

使用道具 举报

82#
ID:871255 发表于 2020-12-30 21:49 | 只看该作者
挺好的,很有借鉴意义,我还是个小菜鸡在单片机方面
回复

使用道具 举报

81#
ID:870994 发表于 2020-12-30 17:33 来自触屏版 | 只看该作者
我的LED也是不亮啊!!!!
回复

使用道具 举报

80#
ID:869640 发表于 2020-12-28 20:58 | 只看该作者
谢谢分享,有借鉴意义
回复

使用道具 举报

79#
ID:869104 发表于 2020-12-28 00:16 | 只看该作者

我的Led 不工作啊
回复

使用道具 举报

78#
ID:829181 发表于 2020-12-26 15:40 | 只看该作者
有帮助
回复

使用道具 举报

77#
ID:830435 发表于 2020-12-3 17:53 | 只看该作者
很好的资料,有助于学习
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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