找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机DS18B20多点测量温度程序与Proteus仿真图

[复制链接]
跳转到指定楼层
楼主
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)


单片机源程序如下:
  1. //头文件包含
  2. #include <AT89X51.H>
  3. #include <Intrins.h>
  4. #include "DS18B20.H"

  5. //引脚定义
  6. sbit                                                         DQ = P2^7;                                                        //数据线端口

  7. //DS18B20序列号,通过调用GetROMSequence()函数在P1口读出(读8次)
  8. const unsigned char code ROMData1[8] = {0x28, 0x33, 0xC5, 0xB8, 0x00, 0x00, 0x00, 0xD7};        //U1
  9. const unsigned char code ROMData2[8] = {0x28, 0x30, 0xC5, 0xB8, 0x00, 0x00, 0x00, 0x8E};        //U2
  10. const unsigned char code ROMData3[8] = {0x28, 0x31, 0xC5, 0xB8, 0x00, 0x00, 0x00, 0xB9};        //U3
  11. const unsigned char code ROMData4[8] = {0x28, 0x32, 0xC5, 0xB8, 0x00, 0x00, 0x00, 0xE0};        //U4
  12. const unsigned char code ROMData5[8] = {0x28, 0x34, 0xC5, 0xB8, 0x00, 0x00, 0x00, 0x52};        //U5
  13. const unsigned char code ROMData6[8] = {0x28, 0x35, 0xC5, 0xB8, 0x00, 0x00, 0x00, 0x65};        //U6
  14. const unsigned char code ROMData7[8] = {0x28, 0x36, 0xC5, 0xB8, 0x00, 0x00, 0x00, 0x3C};        //U7
  15. const unsigned char code ROMData8[8] = {0x28, 0x37, 0xC5, 0xB8, 0x00, 0x00, 0x00, 0x0B};        //U8

  16. //延时16us子函数
  17. void Delay16us()
  18. {
  19.         unsigned char a;

  20.         for (a = 0; a < 4; a++);
  21. }

  22. //延时60us子函数
  23. void Delay60us()
  24. {
  25.         unsigned char a;

  26.         for (a = 0; a < 18; a++);
  27. }

  28. //延时480us子函数
  29. void Delay480us()
  30. {
  31.         unsigned char a;

  32.         for (a = 0; a < 158; a++);
  33. }

  34. //延时240us子函数
  35. void Delay240us()
  36. {
  37.         unsigned char a;

  38.         for (a = 0; a < 78; a++);
  39. }

  40. //延时500ms子函数
  41. void Delay500ms()
  42. {
  43.         unsigned char a, b, c;

  44.         for (a = 0; a < 250; a++)
  45.         for (b = 0; b < 3; b++)
  46.         for (c = 0; c < 220; c++);
  47. }

  48. //芯片初始化
  49. void Initialization()
  50. {
  51.         while(1)
  52.         {
  53.                 DQ = 0;
  54.                 Delay480us();                         //延时480us
  55.                 DQ = 1;
  56.                 Delay60us();                        //延时60us
  57.                 if(!DQ)                                  //收到ds18b20的应答信号
  58.                 {       
  59.                         DQ = 1;
  60.                         Delay240us();                //延时240us
  61.                         break;               
  62.                 }
  63.         }
  64. }

  65. //写一个字节(从低位开始写)
  66. void WriteByte(unsigned char btData)
  67. {
  68.         unsigned char i, btBuffer;

  69.         for (i = 0; i < 8; i++)
  70.         {
  71.                 btBuffer = btData >> i;
  72.                 if (btBuffer & 1)
  73.                 {
  74.                         DQ = 0;
  75.                         _nop_();
  76.                         _nop_();
  77.                         DQ = 1;
  78.                         Delay60us();
  79.                 }
  80.                 else
  81.                 {
  82.                         DQ = 0;
  83.                         Delay60us();
  84.                         DQ = 1;                       
  85.                 }
  86.         }
  87. }

  88. //读一个字节(从低位开始读)
  89. unsigned char ReadByte()
  90. {
  91.         unsigned char i, btDest;

  92.         for (i = 0; i < 8; i++)
  93.         {
  94.                 btDest >>= 1;
  95.                 DQ = 0;
  96.                 _nop_();
  97.                 _nop_();
  98.                 DQ = 1;
  99.                 Delay16us();
  100.                 if (DQ) btDest |= 0x80;
  101.                 Delay60us();
  102.         }

  103.         return btDest;
  104. }

  105. //序列号匹配
  106. void MatchROM(const unsigned char *pMatchData)
  107. {
  108.         unsigned char i;

  109.         Initialization();
  110.         WriteByte(MATCH_ROM);
  111.         for (i = 0; i < 8; i++) WriteByte(*(pMatchData + i));       
  112. }

  113. //得到64位ROM序列(在P1口显示,必须与Proteus联调且在单步调试下才能得到)
  114. /*void GetROMSequence()
  115. {
  116.         unsigned char i;

  117.         Initialization();
  118.         WriteByte(READ_ROM);
  119.         for (i = 0; i < 8; i++)
  120.         P1 = ReadByte();       
  121. }*/

  122. //读取温度值
  123. TEMPDATA ReadTemperature()
  124. {
  125.         TEMPDATA TempData;
  126.         unsigned int iTempDataH;
  127.         unsigned char btDot, iTempDataL;
  128.         static unsigned char i = 0;

  129.         TempData.btNegative = 0;                                                //为0温度为正
  130.         i++;
  131.         if (i == 9) i = 1;
  132.         Initialization();
  133.         WriteByte(SKIP_ROM);                                                        //跳过ROM匹配
  134.         WriteByte(TEMP_SWITCH);                                                        //启动转换
  135.         Delay500ms();                                                                          //调用一次就行       
  136.         Delay500ms();                         
  137.         Initialization();

  138.         //多个芯片的时候用MatchROM(ROMData)换掉WriteByte(SKIP_ROM)
  139.         switch (i)
  140.         {
  141.                 case 1 : MatchROM(ROMData1); break;                        //匹配1
  142.                 case 2 : MatchROM(ROMData2); break;                        //匹配2
  143.                 case 3 : MatchROM(ROMData3); break;                        //匹配3
  144.                 case 4 : MatchROM(ROMData4); break;                        //匹配4       
  145.                 case 5 : MatchROM(ROMData5); break;                        //匹配5
  146.                 case 6 : MatchROM(ROMData6); break;                        //匹配6
  147.                 case 7 : MatchROM(ROMData7); break;                        //匹配7
  148.                 case 8 : MatchROM(ROMData8); break;                        //匹配8
  149.         }
  150.         //WriteByte(SKIP_ROM);                                                        //跳过ROM匹配(单个芯片时用这句换掉上面的switch)
  151.         WriteByte(READ_MEMORY);                                                        //读数据
  152.         iTempDataL = ReadByte();
  153.         iTempDataH = ReadByte();       
  154.         iTempDataH <<= 8;
  155.         iTempDataH |= iTempDataL;

  156.         if (iTempDataH & 0x8000)
  157.         {
  158.                 TempData.btNegative = 1;
  159.                 iTempDataH = ~iTempDataH + 1;                                //负数求补
  160.         }

  161.         //为了省去浮点运算带来的开销,而采用整数和小数部分分开处理的方法(没有四舍五入)
  162.         btDot = (unsigned char)(iTempDataH & 0x000F);        //得到小数部分
  163.         iTempDataH >>= 4;                                                                //得到整数部分
  164.         btDot *= 5;                                                                         //btDot*10/16得到转换后的小数数据
  165.         btDot >>= 3;

  166.         //数据处理
  167.         TempData.btThird   = (unsigned char)iTempDataH / 100;
  168.         TempData.btSecond  = (unsigned char)iTempDataH % 100 / 10;
  169.         TempData.btFirst   = (unsigned char)iTempDataH % 10;
  170.         TempData.btDecimal = btDot;       

  171.         return TempData;
  172. }
复制代码

Keil2代码与Proteus7.5和8.8仿真下载:
多点温度测量.7z (59.97 KB, 下载次数: 49)

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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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