找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机智能温度检测系统源程序,红外遥控控制设定阈值

[复制链接]
跳转到指定楼层
楼主
ID:629845 发表于 2019-12-16 11:40 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
单片机源程序如下:
  1. /*
  2. 实验说明:
  3.         根据自己使用的LCD1602是否带有转接板,如果带有转接板的即为4位,需在LCD.H头文件中
  4.         将宏#define LCD1602_4PINS打开,我们这里使用的LCD1602是8位,所以默认将该宏注释。
  5. 实验接线:
  6.         1,LCD1602液晶模块-->单片机管脚
  7.         参考LCD1602液晶显示实验接线(开发攻略内在对应的实验章节内实验现象有接线说明)
  8.         2,红外遥控模块-->单片机管脚
  9.         P32
  10.         3,蜂鸣器模块-->单片机管脚
  11.         BEEP-->P15
  12.         4,LED模块-->单片机管脚
  13.         D1-->P24
  14.         5,LED模块-->单片机管脚
  15.         D1-->P24
  16.         6,继电器模块-->单片机管脚
  17.         RELAY-->P14
  18.         7,直流电机模块-->单片机管脚
  19.         IN1-->P10(参考直流电机实验接线)
  20.         8,EEPROM模块-->单片机管脚
  21.         P20-->SDA
  22.         P21-->SCL
  23.         9,DS18B20模块-->单片机管脚
  24.         P37
  25.         
  26. 实验现象:
  27.         首先它能显示环境的温度并能设置上下限阀值,这是最基本的功能,系统上电的时候显示的是当前环境温度
  28.         和设定的温度阀值,我们可以通过红外遥控键来修改温度上下限阀值。我们看,按下这个1键会进入温度阀值设置
  29.         界面,每按一下,切换一次阀值设置(上下阀值)界面,按第3次时,会自动回到主界面,如此循环。
  30.         在进入温度阀值设计界面时,可以通过2、3键对阀值进行加减,这里我们只对温度整数部分进行设置,
  31.         小数部分我们就不需要了,将设置好的上下限阀值保存到AT24C02(EEPROM)内,当下一次开启系统时只需
  32.         从AT24C02内读取保存的阀值数据,而不需要重复设置上下限阀值。这样的话,我们用3个按键就实现了温度上下
  33.         限阀值的设定,这是温度检测控制系统基本的功能。假如我们把温度上限设置为32°C,下限设置为30°C。
  34.         另外还有恒定温度的功能。当设定好上下限阀值时,系统即会把当前的温度与设定的上下限阀值对比,
  35.         如果高于上限温度,开启散热进行降温,同时报警;如果低于下限温度,开启加热,同时报警;
  36.         如果当前温度处于下限和上限温度之间时,关闭散热、加热及报警。从而可将温度控制在阀值的范围内。
  37. */

  38. #include "public.h"
  39. #include "lcd.h"
  40. #include "temp.h"
  41. #include "i2c.h"                          

  42. sbit led=P2^4; //报警指示灯

  43. sbit beep=P1^5;          //蜂鸣器报警
  44. sbit relay=P1^4;  //加热设备
  45. sbit moto=P1^0;   //电机散热

  46. sbit IRIN=P3^2;

  47. u8 IrValue[6];
  48. u8 Time,K;
  49. char set_templ=5,set_temph=20;  //设定温度上下限默认值
  50. u16 temp_val;  //检测的实际温度
  51. u8 mode;  //温度模式

  52. u8 irtime;          //接收时间
  53. u8 startflag;  //开始检测
  54. u8 bitnum;          //bit位数
  55. u8 irdata[33]; //寄存没有个bit的时间
  56. u8 irreceok;          //接收完成标志位
  57. u8 ircode[4];  //将接收的四个字节数据保存
  58. u8 irprosok;          //接收数据处理标志位
  59. u8 irdisp[8];  //将四个字节的16进制通过转换变成十进制存储
  60. u8 dat1,dat2;
  61. u8 keynum;
  62. u8 motoflag;

  63. int motopros(u8 mode);

  64. void int0init()
  65. {
  66.         EA=1;
  67.         EX0=1;
  68.         IT0=1;
  69. }
  70. void time0init()
  71. {
  72.         TMOD=0X02;        //设置定时器0模式2.该模式为自动装载模式
  73.         TH0=0X00;
  74.         TL0=0X00;//设定计数初值,每当TL0计数到255时,TH0将把自己的数据给TL0,又重新计数
  75.         TR0=1;
  76.         ET0=1;
  77.         EA=1;        
  78. }
  79. void irpros() //红外接收数据处理 ,区分是数据0还是1
  80. {
  81.         u8 i,j,value;
  82.         u8 k=1;        //引导码去掉,所以令k=1;
  83.         for(j=0;j<4;j++)  //取出了一帧数据中的四个字节数据
  84.         {
  85.                 for(i=0;i<8;i++)        //取出了一个字节的数据               
  86.                 {
  87.                         value>>=1;
  88.                         if(irdata[k]>6)
  89.                         {
  90.                                 value=value|0x80;
  91.                         }
  92.                         k++;
  93.                 }
  94.                 ircode[j]=value;
  95.         }
  96.         irprosok=1;
  97. }
  98. void irwork()  //将四个字节的数据转换成10进制数显示
  99. {
  100.         irdisp[0]=ircode[0]/16;//显示高位
  101.         irdisp[1]=ircode[0]%16;//显示低位  
  102.         irdisp[2]=ircode[1]/16;
  103.         irdisp[3]=ircode[1]%16;         //同一个遥控器此2个字节的引导码数据是不会改变的,改变的只是数据码及反码
  104.         irdisp[4]=ircode[2]/16;
  105.         irdisp[5]=ircode[2]%16;
  106.         irdisp[6]=ircode[3]/16;
  107.         irdisp[7]=ircode[3]%16;
  108.         dat1=irdisp[4];
  109.         dat2=irdisp[5];
  110.         keynum++;
  111. }

  112. void Temp_DataPros()
  113. {
  114.         short temp;
  115.         u8 temp_buf[5];
  116.         temp=Ds18b20ReadTemp();
  117.         temp_val=temp;        
  118.         if(temp<0)
  119.         {
  120.                 temp=-temp;
  121.                 LCD_Dispstring(2+5,0,"-");
  122.         }
  123.         else
  124.         {
  125.                 LCD_Dispstring(2+5,0," ");
  126.         }

  127.         temp_buf[0]=temp/100+0x30;
  128.         temp_buf[1]=temp%100/10+0x30;
  129.         temp_buf[2]='.';
  130.         temp_buf[3]=temp%100%10+0x30;
  131.         temp_buf[4]='\0';
  132.         LCD_Dispstring(2+6,0,temp_buf);                 //显示检测的温度xx.x

  133.         temp_buf[0]=set_temph/10+0x30;
  134.            temp_buf[1]=set_temph%10+0x30;
  135.         temp_buf[2]='\0';
  136.         LCD_Dispstring(5,1,temp_buf);          //显示设定的温度上限值xx

  137.         temp_buf[0]=set_templ/10+0x30;
  138.            temp_buf[1]=set_templ%10+0x30;
  139.         temp_buf[2]='\0';
  140.         LCD_Dispstring(14,1,temp_buf);         //显示设定的温度下限值xx
  141. }

  142. #define  K1_MODE  1
  143. #define  K2_ADD  2
  144. #define  K3_DEC  3

  145. void KEY_Pros()
  146. {
  147.         u8 key;
  148.         u8 temph_buf[3];

  149.         key=motopros(0);
  150.         
  151.         if(key==K1_MODE)   //模式选择
  152.         {
  153.                 mode++;
  154.                 LCD_Clear();

  155.                 if(mode==1)
  156.                 {
  157.                         LCD_Dispstring(0,0,"SETH:   C");               
  158.                 }
  159.                 else if(mode==2)
  160.                 {
  161.                         LCD_Dispstring(0,1,"SETL:   C");        
  162.                 }
  163.                 else
  164.                 {
  165.                         mode=0;
  166.                         LCD_Dispstring(2,0,"Temp:     C");
  167.                         LCD_Dispstring(0,1,"SETH:  ");
  168.                         LCD_Dispstring(9,1,"SETL:  ");        
  169.                 }
  170.         }
  171.         if(mode==1)         //温度上限设置
  172.         {
  173.                 switch(key)        
  174.                 {
  175.                         case K2_ADD:            //加
  176.                                                 set_temph++;
  177.                                                 if(set_temph>=80)set_temph=80;
  178.                                                 break;
  179.                         case K3_DEC:          //减
  180.                                                 set_temph--;
  181.                                                 if(set_temph<=0)set_temph=0;
  182.                                                 break;
  183.                 }

  184.                 temph_buf[0]=set_temph/10+0x30;
  185.                 temph_buf[1]=set_temph%10+0x30;
  186.                 temph_buf[2]='\0';
  187.                    LCD_Dispstring(6,0,temph_buf);
  188.                 At24c02Write(0,set_temph);
  189.         }

  190.         else if(mode==2)  //温度下限设置
  191.         {
  192.                 switch(key)        
  193.                 {
  194.                         case K2_ADD:            //加
  195.                                                 set_templ++;
  196.                                                 if(set_templ>=80)set_templ=80;
  197.                                                 break;
  198.                         case K3_DEC:          //减
  199.                                                 set_templ--;
  200.                                                 if(set_templ<=0)set_templ=0;
  201.                                                 break;
  202.                 }

  203.                 temph_buf[0]=set_templ/10+0x30;
  204.                 temph_buf[1]=set_templ%10+0x30;
  205.                 temph_buf[2]='\0';
  206.                    LCD_Dispstring(6,1,temph_buf);
  207.                 At24c02Write(2,set_templ);
  208.         }
  209. }

  210. void sound()
  211. {
  212.         u8 i=50;
  213.         while(i--)
  214.         {
  215.                 beep=!beep;
  216.                 delay(10);
  217.         }
  218. }
  219. void TempData_Compare()
  220. {
  221.         if(temp_val>set_temph*10)        //实际温度高于上限值   报警  散热
  222.         {
  223.                 led=0;
  224.                 moto=1;
  225.                 relay=0;
  226.                 sound();               
  227.         }
  228.         else if(temp_val<set_templ*10)         //实际温度低于下限值   报警  加热
  229.         {
  230.                 led=0;
  231.                 moto=0;
  232.                 relay=0;
  233.                 sound();               
  234.         }
  235.         else                      //实际温度在下限值和上限值之间   取消报警  取消加热  取消散热
  236.         {
  237.                 moto=0;
  238.                 led=1;
  239.                 relay=1;
  240.         }
  241. }

  242. void kai_display()
  243. {
  244.         if(At24c02Read(255)!=18)
  245.         {
  246.                 At24c02Write(0,set_temph);
  247.                 At24c02Write(2,set_templ);
  248.                 At24c02Write(255,18);        
  249.         }
  250.         else
  251.         {
  252.                 set_temph=At24c02Read(0);
  253.                 set_templ=At24c02Read(2);        
  254.         }

  255.         LCD_Dispstring(2,0,"Temp:     C");
  256.         LCD_Dispstring(0,1,"SETH:  ");
  257.         LCD_Dispstring(9,1,"SETL:  ");
  258. }

  259. //mode:  0-单次扫描   1-连续扫描
  260. int motopros(u8 mode)
  261. {
  262.         static u8 key=1;
  263.         u8 k1,k2,k3;
  264.         if(key)
  265.         {
  266.                 key=0;
  267.                 if((dat1==0)&&(dat2==12))  //按下第一次按键开马达
  268.                 {                        
  269.                         dat1=0;
  270.                         dat2=0;
  271.                         k1=0;
  272.                         return K1_MODE;                                
  273.                 }
  274.                                 
  275.                 if((dat1==1)&&(dat2==8))        //按下第二次此按键时关闭马达
  276.                 {
  277.                         dat1=0;
  278.                         dat2=0;
  279.                         k2=0;
  280.                         return K2_ADD;
  281.                 }
  282.                 if((dat1==5)&&(dat2==14))        //按下第二次此按键时关闭马达
  283.                 {
  284.                         dat1=0;
  285.                         dat2=0;
  286.                         k3=0;
  287.                         return K3_DEC;
  288.                 }
  289.                 if((dat1==0)&&(dat2==8))        //按下第二次此按键时关闭马达
  290.                 {
  291.                         dat1=0;
  292.                         dat2=0;
  293.                         keynum=0;               
  294.                 }
  295.         }

  296.         else if(k1==1&&k2==1&&k3==1)
  297.         {
  298.                 key=1;        
  299.         }
  300.         if(mode)
  301.         {
  302.                 key=1;
  303.         }
  304.         return 0;
  305.         k1=1;k2=1;k3=1;                                                
  306. }

  307. void main()
  308. {
  309.         moto=0;
  310.         led=1;
  311.         relay=1;

  312.         int0init();
  313.         time0init();
  314.         LCD_Init();
  315.         kai_display();
  316.         while(1)
  317.         {
  318.                 if(irreceok)
  319.                 {
  320.                         irreceok=0;
  321.                         irpros();
  322.                         if(irprosok)
  323.                         {
  324.                                 irwork();
  325.                         }
  326.                 }
  327.                 motopros();
  328.                 if(mode==0)
  329.                         Temp_DataPros();
  330.                 KEY_Pros();        
  331.                 TempData_Compare();                                
  332.         }
  333. }

  334. void time0() interrupt 1
  335. {
  336.         irtime++;//每进来一次就说明定时时间为256us;        
  337. }
  338. void init0() interrupt 0
  339. {
  340.         if(startflag)
  341.         {
  342.                 if(irtime>32)//检测引导码,求法是用引导码时间除以一次计数时间,看看要多少次
  343.                 {
  344.                         bitnum=0;        
  345.                 }
  346.                 irdata[bitnum]=irtime;
  347.                 irtime=0;
  348.                 bitnum++;
  349.                 if(bitnum==33)
  350.                 {
  351.                         bitnum=0;
  352.                         irreceok=1;//一帧红外数据接收完成标志
  353.                 }
  354.         }
  355.         else
  356.         {
  357.                 startflag=1;//将开始标志位置1,等到下次进入中断即可进入if语句
  358.                 irtime=0;//将开始之前的计数器时间清零。等到下次进入中断的时候才是引导码真正的时间
  359.         }
  360. }
复制代码

所有程序51hei提供下载:
智能温度检测控制系统设计.rar (72.88 KB, 下载次数: 51)

评分

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

查看全部评分

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

使用道具 举报

沙发
ID:583775 发表于 2019-12-22 14:27 | 只看该作者
没有原理图
回复

使用道具 举报

板凳
ID:629845 发表于 2019-12-27 20:47 | 只看该作者

我是用的普中51单片机开发板,可以把开发板的原理图给你,没有画仿真电路图
回复

使用道具 举报

地板
ID:629845 发表于 2019-12-27 20:49 | 只看该作者
用的开发板,我把开发板原理图给你

普中51-双核-A6开发板原理图.pdf

205.83 KB, 下载次数: 12, 下载积分: 黑币 -5

回复

使用道具 举报

5#
ID:861537 发表于 2020-12-16 17:00 | 只看该作者
你好,这个是按键控制还是红外控制啊,为什么设置温度加减时定义的是K2和K3
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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