标题: 2051单片机DS18B20数字温度计C程序 [打印本页]

作者: dzdiy    时间: 2015-4-11 20:04
标题: 2051单片机DS18B20数字温度计C程序
      这个温度计使用的单片机是AT89C2051,测量范围-55到+125度,显示精度为0.1度,DS18B20数据脚P3.4 ,采用器件默认的12位转化,最大转化时间750微秒,显示采用4位LED共阳显示测温值,P1口为段码输入,P0~P3为位选。下面是制作好实物图,电路原理比较简单,这里就不给出。资料是从网络上收集整理的。      

源代码下载: DS18B20数字温度计C程序.zip (22.71 KB, 下载次数: 283)




程序源代码:
  1. //***********DS18B20数字温度计C程序****************//
  2. //*MCU:          AT892051                          //
  3. //*MCU-crystal:  12M                               //
  4. //DS18B20的读写程序,数据脚P3.4                     //
  5. //温度传感器18B20汇编程序,采用器件默认的12位转化   //
  6. //最大转化时间750微秒,显示温度-55到+125度,显示精度 //
  7. //为0.1度,显示采用4位LED共阳显示测温值            //
  8. //P1口为段码输入,P0~P3为位选                       //
  9. /***************************************************/

  10. #include "reg51.h"
  11. #include "intrins.h"                            //_nop_();延时函数用
  12. #define  Disdata P1                             //段码输出口
  13. #define  discan  P3                             //扫描口
  14. #define  uchar unsigned char
  15. #define  uint  unsigned int
  16. sbit     DQ=P3^4;                               //温度输入口
  17. sbit     DIN=P1^7;                              //LED小数点控制
  18. uint     h;
  19. uint     temp;
  20. //
  21. //
  22. //**************温度小数部分用查表法***********//
  23. uchar code ditab[16]=
  24. {0x00,0x01,0x01,0x02,0x03,0x03,0x04,0x04,0x05,0x06,0x06,0x07,0x08,0x08,0x09,0x09};
  25. //
  26. uchar code dis_7[12]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xff,0xbf};
  27. //共阳LED段码表        "0"  "1"  "2"  "3"  "4"  "5"  "6"  "7"  "8"  "9" "不亮" "-"              
  28. uchar code scan_con[4]={0x08,0x04,0x02,0x01};    //列扫描控制字
  29. uchar data temp_data[2]={0x00,0x00};             //读出温度暂放
  30. uchar data display[5]={0x00,0x00,0x00,0x00,0x00};     //显示单元数据,共4个数据和一个运算暂用

  31. //
  32. //
  33. //
  34. /*****************11us延时函数*************************/
  35. //
  36. void delay(uint t)
  37. {
  38.   for (;t>0;t--);
  39. }
  40. //

  41. /****************显示扫描函数***************************/
  42. scan()
  43. {
  44.   char k;
  45.      for(k=0;k<4;k++)                     //4位LED扫描控制
  46.   {
  47.    Disdata=dis_7[display[k]];          //数据显示
  48.    if (k==1){DIN=0;}                   //小数点显示
  49.    discan=scan_con[k];                 //位选
  50.    delay(300);
  51.   }
  52.   return 0;
  53. }
  54. //
  55. //
  56. /****************DS18B20复位函数************************/
  57. ow_reset(void)
  58. {
  59. char presence=1;
  60. while(presence)
  61. {
  62.   while(presence)
  63.    {
  64.     DQ=1;_nop_();_nop_();//从高拉倒低
  65. DQ=0;                                       
  66. delay(50);           //550 us
  67. DQ=1;                                       
  68. delay(6);            //66 us
  69. presence=DQ;         //presence=0 复位成功,继续下一步
  70.    }
  71.    delay(45);            //延时500 us
  72.    presence=~DQ;         
  73. }
  74. DQ=1;                   //拉高电平
  75. return 0;
  76. }
  77. //
  78. //
  79. /****************DS18B20写命令函数************************/
  80. //向1-WIRE 总线上写1个字节
  81. void write_byte(uchar val)
  82. {
  83.   uchar i;
  84.   for(i=8;i>0;i--)
  85.   {
  86.    DQ=1;_nop_();_nop_();                  //从高拉倒低
  87.    DQ=0;_nop_();_nop_();_nop_();_nop_();  //5 us
  88.    DQ=val&0x01;                           //最低位移出
  89.    delay(6);                              //66 us
  90.    val=val/2;                             //右移1位
  91.    }
  92.    DQ=1;
  93.    delay(1);
  94. }
  95. //
  96. /****************DS18B20读1字节函数************************/
  97. //从总线上取1个字节
  98. uchar read_byte(void)
  99. {
  100. uchar i;
  101. uchar value=0;
  102. for(i=8;i>0;i--)
  103. {
  104.   DQ=1;_nop_();_nop_();
  105.   value>>=1;
  106.   DQ=0;_nop_();_nop_();_nop_();_nop_();         //4 us
  107.   DQ=1;_nop_();_nop_();_nop_();_nop_();         //4 us
  108.   if(DQ)value|=0x80;
  109.   delay(6);                                     //66 us
  110. }
  111. DQ=1;
  112. return(value);
  113. }
  114. //
  115. /****************读出温度函数************************/
  116. //
  117. read_temp()
  118. {
  119.   ow_reset();                  //总线复位
  120.   delay(200);
  121.   write_byte(0xcc);            //发命令
  122.   write_byte(0x44);            //发转换命令
  123.   ow_reset();
  124.   delay(1);
  125.   write_byte(0xcc);            //发命令
  126.   write_byte(0xbe);
  127.   temp_data[0]=read_byte();    //读温度值的第字节
  128.   temp_data[1]=read_byte();    //读温度值的高字节
  129.   temp=temp_data[1];
  130.   temp<<=8;                                               
  131.   temp=temp|temp_data[0];      // 两字节合成一个整型变量。
  132.   return temp;                 //返回温度值
  133. }
  134. //
  135. /****************温度数据处理函数************************/

  136. //二进制高字节的低半字节和低字节的高半字节组成一字节,这个
  137. //字节的二进制转换为十进制后,就是温度值的百、十、个位值,而剩
  138. //下的低字节的低半字节转化成十进制后,就是温度值的小数部分

  139. /********************************************************/
  140. work_temp(uint tem)
  141. {
  142. uchar n=0;
  143. if(tem>6348)                        // 温度值正负判断
  144.      {tem=65536-tem;n=1;}            // 负温度求补码,标志位置1
  145.       display[4]=tem&0x0f;           // 取小数部分的值
  146.    display[0]=ditab[display[4]];  // 存入小数部分显示值
  147.       display[4]=tem>>4;             // 取中间八位,即整数部分的值
  148.    display[3]=display[4]/100;     // 取百位数据暂存
  149.    display[1]=display[4]%100;     // 取后两位数据暂存
  150.    display[2]=display[1]/10;      // 取十位数据暂存
  151.    display[1]=display[1]%10;  
  152. /******************符号位显示判断**************************/
  153.   if(!display[3])
  154.   {
  155.     display[3]=0x0a;           //最高位为0时不显示
  156.     if(!display[2])
  157.   {
  158.     display[2]=0x0a;        //次高位为0时不显示
  159.   }
  160.   }
  161.   if(n){display[3]=0x0b;}      //负温度时最高位显示"-"
  162.   return 0;
  163. }
  164. //
  165. //
  166. /****************主函数************************/
  167. main()
  168. {
  169.   Disdata=0xff;                 //初始化端口
  170.   discan=0xff;
  171.   for(h=0;h<4;h++)              //开机显示"8888"
  172.       {display[h]=8;}
  173.   ow_reset();                   //开机先转换一次
  174.   write_byte(0xcc);             //Skip ROM
  175.   write_byte(0x44);             //发转换命令
  176.   for(h=0;h<100;h++)            //开机显示"8888"
  177.       {scan();}
  178.   while(1)
  179.   {
  180. work_temp(read_temp());     //处理温度数据
  181. scan();                     //显示温度值
  182.   }
  183. }
  184. //
  185. //***********************结束**************************//
复制代码




作者: kmtbm    时间: 2015-5-26 19:54
智能热水器控制系统
作者: lydjd    时间: 2015-7-29 09:44
小巧,简单,不错。
作者: libo13990    时间: 2015-10-2 00:55
新手过来学习学习
作者: moonchen    时间: 2016-5-3 16:09
很好,适合新手练手。
作者: woshilco123    时间: 2016-6-1 15:41
看看,还不错的样子
作者: gdx    时间: 2016-6-22 14:28
过来学习学习
作者: 勇敢的心oill    时间: 2016-8-6 10:08
能不能提供个原理图啊,新手
作者: 勇敢的心oill    时间: 2016-8-6 10:09
gdx 发表于 2016-6-22 14:28
过来学习学习

你有适合这个程序的原理图吗
作者: 勇敢的心oill    时间: 2016-8-6 10:09
moonchen 发表于 2016-5-3 16:09
很好,适合新手练手。

你有适合这个程序的原理图吗
作者: 越快乐越堕落    时间: 2016-8-6 13:56
冒昧说一句,18B20不好控制,楼主给出的图和程序就算全部都能对的上号,楼主也做成功了,但是不见得别人就能一样成功完美的工作,因为这货的程序要求很严格,还有硬件也会有误差,所以慎重选择。
我用这货反正是没成功过,连毛都不显示,显示出来的也没人看得懂。。。
作者: wc86110    时间: 2016-8-6 16:39
越快乐越堕落 发表于 2016-8-6 13:56
冒昧说一句,18B20不好控制,楼主给出的图和程序就算全部都能对的上号,楼主也做成功了,但是不见得别人就 ...

同样可用的温度计制作,不见得人人都能制作成功。
DIY 还是要看个人能力

作者: 越快乐越堕落    时间: 2016-8-7 17:47
wc86110 发表于 2016-8-6 16:39
同样可用的温度计制作,不见得人人都能制作成功。
DIY 还是要看个人能力

个人能力是一方面,元件性能也是关键。
作者: zonafrist    时间: 2016-10-15 21:40
学习学习,自己的程序老有问题,参考楼主的,谢谢
作者: z1356504825    时间: 2016-11-15 09:26
感谢楼主
作者: zyh287278    时间: 2016-12-15 23:07
很好,学习学习
作者: nenner    时间: 2016-12-22 19:20
不错的程序
作者: YwJ    时间: 2017-1-21 07:56
很好,学习学习
作者: 一个童话    时间: 2017-3-22 19:26
谁有这程序原理图,我怎么链接完显示很乱,谢谢大家了
作者: lyf898    时间: 2017-4-3 13:57
确实够简洁,初学者很有用
作者: XDDZ1013    时间: 2017-6-9 21:49
谢谢分享
作者: 348652560    时间: 2017-6-14 22:54
谢谢分享,太好了正好需要
作者: juxun001    时间: 2017-12-11 14:45
能给我个资料吗540371974@qq.com
作者: microhl    时间: 2019-9-25 10:12
不知为何,温度显示不变
作者: microhl    时间: 2019-9-25 10:14
这个可以深刻理解I2C的工作原理
作者: 15071173583    时间: 2019-9-29 09:25
赞,,真不错




欢迎光临 (http://www.51hei.com/bbs/) Powered by Discuz! X3.1