单片机论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

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

ATMEGA16A单片机的多点测温系统完成版 Proteus仿真程序

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


单片机源程序如下:
  1. /**************************************************************************
  2. *实验名   :DS18B20模块
  3. *实验效果 :单总线调试,此代码可用于测多点温度
  4. *显示     :此代码用了LCD显示读取地址位
  5. *平台     : Atmega16,基于8MHz
  6. *调试时间 :2019年6月7日 23:14
  7. **************************************************************************/
  8. #include <iom16v.h>
  9. #include <macros.h>
  10. #include "ds18b20.h"
  11. //#include "lcd1602.h"
  12. //微妙级延迟函数avr 16 以8MHz为例加上for循环,他需要执行1142条指令才是1ms,
  13. void delay_1us(void)
  14. {
  15.   asm("nop");
  16. }
  17. void Delay_1us(unsigned int x)//所以这里存在一定的偏差。
  18. {
  19.    unsigned int i=0;     
  20.    for( i=0;i<x;i++)delay_1us();
  21. }
  22. //DS18B20复位函数
  23. void DS18B20_Reset(void)
  24. { /*
  25.     DQ_1;   //输出低电平
  26. DQ_OUT;   //DQ为输出状态
  27. DQ_0;   //输出低电平
  28. Delay_1us(500);   //延迟500微妙
  29. DQ_1;    //释放总线
  30. Delay_1us(60);   //延迟60微妙
  31. DQ_IN;   //DQ位输入状态
  32. while(DQ_RD); //等待从机DS18B20应答(低电平有效)
  33. while(!(DQ_RD));*/ //等待从机DS18B20释放总线机DS18B20释放总线
  34.     //这里要加个括号判断优先级
  35.   while(1)
  36. {
  37.        DQ_OUT;
  38.     DQ_0;
  39.     Delay_1us(480);   //延时480us
  40.     DQ_1;
  41.     Delay_1us(60);
  42.     DQ_IN;  //设置为输入端,接收应答信号
  43.     if(!(DQ_RD))      //收到应答信号
  44.    {
  45.   DQ_1;
  46.   Delay_1us(240);
  47.   break;
  48.    }  //延时240us
  49. }  
  50. }
  51. //DS18B20写字节函数
  52. void DS18B20_Write(unsigned char Data)
  53. {
  54. unsigned char i;
  55. DQ_OUT;  //DQ为输出
  56. for(i=0;i<8;i++)
  57. {
  58.   DQ_0;   //拉低总线
  59.   Delay_1us(10);     //延迟10微妙(最大15微妙)
  60.   if(Data&0x01)DQ_1;
  61.   else DQ_0;   
  62.   Delay_1us(40);      //延迟40微妙(最大45微妙)
  63.   DQ_1;     //释放总线
  64.   Delay_1us(1);     //稍微延迟
  65.   Data>>=1;
  66. }
  67. }
  68. //DS18B20读字节函数
  69. unsigned char DS18B20_Read(void)
  70. {
  71. unsigned char i,Temp;         
  72. for(i=0;i<8;i++)
  73. {
  74.     Temp>>=1;      //数据右移
  75.     DQ_OUT;     //DQ为输出状态
  76.     DQ_0;    //拉低总线,启动输入
  77.     DQ_1;     //释放总线
  78.     DQ_IN;     //DQ为输入状态
  79.     Delay_1us(2);
  80.     if(DQ_RD) Temp|=0x80;
  81.     Delay_1us(60);      //延迟45微妙(最大45微妙)
  82. }

  83. return Temp;
  84. }
  85. /*void GetROMSequence()//读地址号代码,用lcd1602显示
  86. {
  87. unsigned char i,temp;
  88. DS18B20_Reset();
  89. DS18B20_Write(READ_ROM);
  90. for (i = 0; i < 8; i++)
  91. ROMData1[i] = DS18B20_Read();
  92. lcd_com(0x80);   
  93. for(i=0;i<8;i++)
  94.     {
  95.        temp=ROMData1[i]>>4;
  96.        if(temp<10)
  97.        {
  98.          lcd_dat(0x30+temp);
  99.        }
  100.        else
  101.        {
  102.          lcd_dat(0x37+temp);
  103.        }
  104.        temp=ROMData1[i]&0x0f;
  105.        if(temp<10)
  106.        {
  107.          lcd_dat(0x30+temp);
  108.        }
  109.        else
  110.        {
  111.          lcd_dat(0x37+temp);
  112.        }
  113.     }
  114. }*/
  115. //读温度,然后拆字用结构体存起来。
  116. TEMPDATA ReadTemperature(const unsigned char *pMatchData)
  117. {
  118. TEMPDATA TempData;
  119. unsigned int iTempDataH;
  120. unsigned char btDot, iTempDataL;
  121. static unsigned char i = 0;
  122. TempData.btNegative = 0;      //为0温度为正

  123. DS18B20_Reset();   //DS18B20复位
  124. DS18B20_Write(SKIP_ROM);  //跳过ROM
  125. DS18B20_Write(TEMP_SWITCH);  //温度转换

  126. DS18B20_Reset();   //DS18B20复位
  127. DS18B20_Write(MATCH_ROM);//读取地址
  128. for (i = 0; i < 8; i++) DS18B20_Write(*(pMatchData + i));
  129. //DS18B20_Write(SKIP_ROM);  //跳过ROM,单个读取直接跳过ROM
  130. DS18B20_Write(READ_MEMORY);  //读取RAM      //读数据

  131. iTempDataL = DS18B20_Read();
  132. iTempDataH = DS18B20_Read();
  133. DS18B20_Reset(); //读取数值完要复位,要不然读取不了数值
  134. iTempDataH <<= 8;
  135. iTempDataH |= iTempDataL;
  136. if (iTempDataH & 0x8000)
  137. {
  138.   TempData.btNegative = 1;
  139.   iTempDataH = ~iTempDataH + 1;    //负数求补
  140. }
  141. //为了省去浮点运算带来的开销,而采用整数和小数部分分开处理的方法(没有四舍五入)
  142. btDot = (unsigned char)(iTempDataH & 0x000F); //得到小数部分
  143. iTempDataH >>= 4;        //得到整数部分
  144. btDot *= 5;          //btDot*10/16得到转换后的小数数据
  145. btDot >>= 3;
  146. //数据处理
  147. TempData.btThird   = (unsigned char)iTempDataH / 100;
  148. TempData.btSecond  = (unsigned char)iTempDataH % 100 / 10;
  149. TempData.btFirst   = (unsigned char)iTempDataH % 10;
  150. TempData.btDecimal = btDot;
  151. return TempData;
  152. }
复制代码

全部资料51hei下载地址:
基于ATMEGA16A多点测温系统完成版.rar (621.51 KB, 下载次数: 9)

评分

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

查看全部评分

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

使用道具 举报

沙发
51hei团团 发表于 2019-6-29 16:20 | 只看该作者
好资料,51黑有你更精彩!!!
回复

使用道具 举报

板凳
W_Cartman 发表于 2019-7-18 15:17 | 只看该作者
感谢楼主分享!
回复

使用道具 举报

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

本版积分规则

QQ|手机版|小黑屋|单片机论坛 |51Hei单片机16群 联系QQ:125739409;技术交流QQ群7344883

Powered by 单片机教程网

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