找回密码
 立即注册

QQ登录

只需一步,快速开始

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

51单片机DS18b20红外线遥控温度计传感器

[复制链接]
跳转到指定楼层
楼主
这个能实现用红外遥控,输入最高温度最低温度,当大于最高温度或者小于最低温度时, 蜂鸣器报警。
这个程序可以实现用红外线遥控器设定一个大于等于0,小于100 度的整数温度值,当检测的温度大于设定的温度时,蜂鸣器响


单片机源程序如下:


  1. #include "reg52.h"                         
  2. #include"temp.h"       

  3. typedef unsigned int u16;          //对数据类型进行声明定义
  4. typedef unsigned char u8;

  5. sbit LSA = P2 ^ 2;                   //数码管引脚
  6. sbit LSB = P2 ^ 3;                   //数码管引脚
  7. sbit LSC = P2 ^ 4;                   //数码管引脚
  8. sbit beep = P1 ^ 5;                   //蜂鸣器引脚
  9. sbit IRIN = P3 ^ 2;                   //红外传感器引脚

  10. u8 IrValue[6];                  //红外码
  11. u8 Time;                          //红外高电平时间
  12. u16 temp1 = 0;            //预设最高温度
  13. u16 temp2 = 0;            //预设最低温度
  14. u16 sethigh = 1;          //是否在设置最高温度
  15. u8 DisplayData1[8];              //预设温度
  16. u16 change;                      //按键是否被按下


  17. char num = 0;
  18. u8 DisplayData[8];
  19. u8 code smgduan[10] = { 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f };


  20. void delay(u16 i)         //延时10um
  21. {
  22.         while (i--);
  23. }

  24. void datapros(int temp) //处理从温度传感器读入的数值,转换为摄氏度,temp = 温度*100,DisplayData[] 数组中是temp的显示码          
  25. {
  26.         float tp;
  27.         if (temp < 0)                                //当温度值为负数
  28.         {
  29.                 DisplayData[0] = 0x40;           //   -                 因为读取的温度是实际温度的补码,所以减1,再取反求出原码

  30.                 temp = temp - 1;
  31.                 temp = ~temp;
  32.                 tp = temp;
  33.                 temp = tp * 0.0625 * 100 + 0.5;
  34.                 //留两个小数点就*100,+0.5是四舍五入,因为C语言浮点数转换为整型的时候把小数点
  35.                 //后面的数自动去掉,不管是否大于0.5,而+0.5之后大于0.5的就是进1了,小于0.5的就
  36.                 //算加上0.5,还是在小数点后面。

  37.         }
  38.         else
  39.         {
  40.                 DisplayData[0] = 0x00;
  41.                 tp = temp;//因为数据处理有小数点所以将温度赋给一个浮点型变量
  42.                 //如果温度是正的那么,那么正数的原码就是补码它本身
  43.                 temp = tp * 0.0625 * 100 + 0.5;
  44.         }
  45.         if (temp >= temp1 * 100 || temp <= temp2 * 100)
  46.                 beep = ~beep;

  47.         DisplayData[1] = smgduan[temp / 10000];
  48.         DisplayData[2] = smgduan[temp % 10000 / 1000];
  49.         DisplayData[3] = smgduan[temp % 1000 / 100] | 0x80;
  50.         DisplayData[4] = smgduan[temp % 100 / 10];
  51.         DisplayData[5] = smgduan[temp % 10];
  52. }

  53. void DigDisplay()                        //显示 DisplayData[] 中的内容
  54. {
  55.         u8 i;
  56.         for (i = 0; i < 6; i++)
  57.         {
  58.                 switch (i)                             //位选,选择点亮的数码管,
  59.                 {
  60.                 case(0):
  61.                         LSA = 0; LSB = 0; LSC = 0; break;//显示第0位
  62.                 case(1):
  63.                         LSA = 1; LSB = 0; LSC = 0; break;//显示第1位
  64.                 case(2):
  65.                         LSA = 0; LSB = 1; LSC = 0; break;//显示第2位
  66.                 case(3):
  67.                         LSA = 1; LSB = 1; LSC = 0; break;//显示第3位
  68.                 case(4):
  69.                         LSA = 0; LSB = 0; LSC = 1; break;//显示第4位
  70.                 case(5):
  71.                         LSA = 1; LSB = 0; LSC = 1; break;//显示第5位       
  72.                 }
  73.                 P0 = DisplayData[5 - i];             //发送数据
  74.                 delay(100);                          //间隔一段时间扫描       
  75.                 P0 = 0x00;                           //消隐
  76.         }
  77. }

  78. void IrInit()                              //初始化红外传感器
  79. {
  80.         IT0 = 1;                               //下降沿触发
  81.         EX0 = 1;                               //EXO = 1 打开红外线中断0允许,数据在IrValue[2]中
  82.         EA = 1;                                       //打开总中断

  83.         IRIN = 1;                              //初始化端口
  84. }

  85. void DigDisplay1()                           //刚开始温度设置时,温度的显示
  86. {
  87.         u8 i;
  88.         for (i = 0; i < 2; i++)
  89.         {
  90.                 switch (i)                                 //位选,选择点亮的数码管,
  91.                 {
  92.                 case(0):
  93.                         LSA = 0; LSB = 0; LSC = 0; break;//显示第0位
  94.                 case(1):
  95.                         LSA = 1; LSB = 0; LSC = 0; break;//显示第1位       
  96.                 }
  97.                 P0 = DisplayData1[1 - i];            //发送数据,先显示个位,是DisplayData1[1]
  98.                 delay(100);                          //间隔一段时间扫描       
  99.                 P0 = 0x00;                           //消隐
  100.         }
  101. }


  102. void main()                                 
  103. {
  104.         u16 displaytemp;                         //在设置温度时,要显示的值

  105.         IrInit();
  106.         while (1)                                //最高最低温度设置
  107.         {
  108.                 if (IrValue[2] == 0x44)
  109.                 {
  110.                         EX0 = 0;                         //关闭中断0允许,用于温度设置完成后,不在接受中断
  111.                         break;
  112.                 }
  113.                 if (IrValue[2] == 0x07)
  114.                 {
  115.                         sethigh = 0;
  116.                 }
  117.                 if (temp1 >= 0 && temp1 < 9 && sethigh == 1)   //最高温度设置
  118.                 {
  119.                         if (change == 1)
  120.                         {
  121.                                 switch (IrValue[2])
  122.                                 {
  123.                                 case(0x16):
  124.                                         temp1 = temp1 * 10 + 0; break;
  125.                                 case(0x0c):
  126.                                         temp1 = temp1 * 10 + 1; break;
  127.                                 case(0x18):
  128.                                         temp1 = temp1 * 10 + 2; break;
  129.                                 case(0x5e):
  130.                                         temp1 = temp1 * 10 + 3; break;
  131.                                 case(0x08):
  132.                                         temp1 = temp1 * 10 + 4; break;
  133.                                 case(0x1c):
  134.                                         temp1 = temp1 * 10 + 5; break;
  135.                                 case(0x5a):
  136.                                         temp1 = temp1 * 10 + 6; break;
  137.                                 case(0x42):
  138.                                         temp1 = temp1 * 10 + 7; break;
  139.                                 case(0x52):
  140.                                         temp1 = temp1 * 10 + 8; break;
  141.                                 case(0x4a):
  142.                                         temp1 = temp1 * 10 + 9; break;
  143.                                 default:
  144.                                         break;
  145.                                 }
  146.                                 change = 0;
  147.                         }
  148.                         displaytemp = temp1;
  149.                 }
  150.                 if (temp2 >= 0 && temp2 < 9 && sethigh == 0)   //最低温度设置
  151.                 {
  152.                         if (change == 1)
  153.                         {
  154.                                 switch (IrValue[2])
  155.                                 {
  156.                                 case(0x16):
  157.                                         temp2 = temp2 * 10 + 0; break;
  158.                                 case(0x0c):
  159.                                         temp2 = temp2 * 10 + 1; break;
  160.                                 case(0x18):
  161.                                         temp2 = temp2 * 10 + 2; break;
  162.                                 case(0x5e):
  163.                                         temp2 = temp2 * 10 + 3; break;
  164.                                 case(0x08):
  165.                                         temp2 = temp2 * 10 + 4; break;
  166.                                 case(0x1c):
  167.                                         temp2 = temp2 * 10 + 5; break;
  168.                                 case(0x5a):
  169.                                         temp2 = temp2 * 10 + 6; break;
  170.                                 case(0x42):
  171.                                         temp2 = temp2 * 10 + 7; break;
  172.                                 case(0x52):
  173.                                         temp2 = temp2 * 10 + 8; break;
  174.                                 case(0x4a):
  175.                                         temp2 = temp2 * 10 + 9; break;
  176.                                 default:
  177.                                         break;
  178.                                 }
  179.                                 change = 0;
  180.                         }
  181.                         displaytemp = temp2;
  182.                 }
  183.                 DisplayData1[0] = smgduan[displaytemp / 10];   //取十位
  184.                 DisplayData1[1] = smgduan[displaytemp % 10];   //取个位
  185.                 DigDisplay1();

  186.         }

  187.         while (1)                            //检测到的温度显示
  188.         {
  189.                 datapros(Ds18b20ReadTemp());         //数据处理函数
  190.                 DigDisplay();                    //数码管显示函数               
  191.         }

  192. }

  193. void ReadIr() interrupt 0                //红外线的中断函数,用于接收红外信号,把数据存入IrValue[2]中
  194. {
  195.         u8 j, k;
  196.         u16 err;
  197.         Time = 0;
  198.         delay(700);                                 //7ms
  199.         if (IRIN == 0)                                     //确认是否真的接收到正确的信号
  200.         {

  201.                 err = 1000;                                //1000*10us=10ms,超过说明接收到错误的信号
  202.                 /*当两个条件都为真是循环,如果有一个条件为假的时候跳出循环,免得程序出错的时
  203.                 侯,程序死在这里*/

  204.                 while ((IRIN == 0) && (err > 0))             //等待前面9ms的低电平过去                 
  205.                 {
  206.                         delay(1);
  207.                         err--;
  208.                 }

  209.                 if (IRIN == 1)                                    //如果正确等到9ms低电平
  210.                 {
  211.                         err = 500;
  212.                         while ((IRIN == 1) && (err > 0))   //等待4.5ms的起始高电平过去
  213.                         {
  214.                                 delay(1);
  215.                                 err--;
  216.                         }


  217.                         for (k = 0; k < 4; k++)                //共有4组数据
  218.                         {
  219.                                 for (j = 0; j < 8; j++)        //接收一组数据
  220.                                 {

  221.                                         err = 60;
  222.                                         while ((IRIN == 0) && (err > 0))//等待信号前面的560us低电平过去
  223.                                         {
  224.                                                 delay(1);
  225.                                                 err--;
  226.                                         }
  227.                                         err = 500;
  228.                                         while ((IRIN == 1) && (err > 0))         //计算高电平的时间长度。
  229.                                         {
  230.                                                 delay(10);         //0.1ms
  231.                                                 Time++;
  232.                                                 err--;
  233.                                                 if (Time > 30)
  234.                                                 {
  235.                                                         return;
  236.                                                 }
  237.                                         }
  238.                                         IrValue[k] >>= 1;         //k表示第几组数据
  239.                                         if (Time >= 8)                        //如果高电平出现大于565us,那么是1
  240.                                         {
  241.                                                 IrValue[k] |= 0x80;
  242.                                         }
  243.                                         Time = 0;                //用完时间要重新赋值                                                       
  244.                                 }
  245.                         }
  246.                 }

  247.                 if (IrValue[2] != ~IrValue[3])
  248.                 {
  249.                         return;
  250.                 }
  251.                 change = 1;          //有有效的按键被按下

  252.         }
  253. }

复制代码

所有资料51hei提供下载:
tenperature.rar (46.48 KB, 下载次数: 21)




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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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