单片机中使用DS18B20温度传感器C语言程序(参考1)

发布者:Xingfu8888最新更新时间:2016-09-21 来源: eefocus关键字:单片机  DS18B20  温度传感器 手机看文章 扫描二维码
随时随地手机看文章
单片机中使用DS18B20温度传感器C语言程序(参考1)

/********************************************************************************
                               DS18B20 测温程序

    硬件:AT89S52 
    (1)单线ds18b20接 P2.2
    (2)七段数码管接P0口
    (3)使用外部电源给ds18b20供电,没有使用寄生电源 
    软件:Kei uVision 3
    
**********************************************************************************/
#include "reg52.h"
#include "intrins.h"
#define uchar unsigned char
#define uint unsigned int
sbit ds=P2^2;
sbit dula=P2^6;
sbit wela=P2^7;
uchar flag ;
uint temp;               //参数temp一定要声明为 int 型 
uchar code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,
0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};     //不带小数点数字编码

uchar code table1[]={0xbf,0x86,0xdb,0xcf,0xe6,0xed,0xfd,
0x87,0xff,0xef};        //带小数点数字编码

/*延时函数*/
void TempDelay (uchar us)
{
   while(us--);
}

void delay(uint count) //延时子函数
{
uint i;
while(count)
{
i=200;
while(i>0)
i--;
count--;
}
}

/*串口初始化,波特率9600,方式1 */
void init_com()
{
   TMOD=0x20;      //设置定时器1为模式2
   TH1=0xfd;       //装初值设定波特率
   TL1=0xfd;
   TR1=1;          //启动定时器
   SM0=0;          //串口通信模式设置
   SM1=1;
// REN=1;          //串口允许接收数据
   PCON=0;         //波特率不倍频
// SMOD=0;       //波特率不倍频
// EA=1;          //开总中断
//ES=1;          //开串行中断
}

/*数码管的显示 */
void display(uint temp)
{
    uchar bai,shi,ge;
    bai=temp/100;
    shi=temp%100/10;
    ge=temp%100%10;

dula=0;
    P0=table[bai]; //显示百位
dula=1;         //从0到1,有个上升沿,解除锁存,显示相应段 
dula=0;         //从1到0再次锁存
          
    wela=0;
    P0=0xfe;
wela=1;
wela=0;
delay(1);    //延时约2ms

P0=table1[shi]; //显示十位
dula=1;
    dula=0;
    
P0=0xfd;
wela=1;
    wela=0;
    delay(1);

    P0=table[ge]; //显示个位
dula=1;
    dula=0;
    
P0=0xfb;
wela=1;
    wela=0;
    delay(1);
}
/***************************************** 
时序:初始化时序、读时序、写时序。
所有时序都是将主机(单片机)作为主设备,单总
线器件作为从设备。而每一次命令和数据的传输
都是从主机主动启动写时序开始,如果要求单总
线器件回送数据,在进行写命令后,主机需启动
读时序完成数据接收。数据和命令的传输都是低
位在先。 
初始化时序:复位脉冲 存在脉冲
        读;1 或 0时序
        写;1 或 0时序
只有存在脉冲信号是从18b20(从机)发出的,其
它信号都是由主机发出的。
存在脉冲:让主机(总线)知道从机(18b20)已
经做好了准备。
******************************************/
/*--------------------------------------------------------------------------------------------------------------------
初始化:检测总线控制器发出的复位脉冲
和ds18b20的任何通讯都要从初始化开始

初始化序列包括一个由总线控制器发出的复位脉冲
和跟在其后由从机发出的存在脉冲。

初始化:复位脉冲+存在脉冲

具体操作:
总线控制器发出(TX)一个复位脉冲 (一个最少保持480μs 的低电平信号),然后释放总线,
进入接收状态(RX)。单线总线由5K 上拉电阻拉到高电平。探测到I/O 引脚上的上升沿后
DS1820 等待15~60μs,然后发出存在脉冲(一个60~240μs 的低电平信号)。

具体看18b20 单线复位脉冲时序和1-wire presence detect "的时序图
-------------------------------------------------------------------------------------------------------------------*/
void ds_reset(void)
{
   ds=1;
   _nop_();        //1us
   ds=0;
   TempDelay(80); //当总线停留在低电平超过480us,总线上所以器件都将被复位,这里//延时约530us总    线停留在低电平超过480μs,总线上的所有器件都
//将被复位。
   _nop_(); 
   ds=1;           //产生复位脉冲后,微处理器释放总线,让总线处于空闲状态,原因查//18b20中文资料

   TempDelay(5); //释放总线后,以便从机18b20通过拉低总线来指示其是否在线,
                  //存在检测高电平时间:15~60us, 所以延时44us,进行            1-wire presence //detect(单线存在检测)
   _nop_();
   _nop_();
   _nop_();
   if(ds==0)
        flag=1;       //detect 18b20 success
   else
        flag=0;       //detect 18b20 fail
TempDelay(20);    //存在检测低电平时间:60~240us,所以延时约140us
   _nop_();
   _nop_();
   ds=1;          //再次拉高总线,让总线处于空闲状态
/**/
}

/*----------------------------------------
读/写时间隙:
DS1820 的数据读写是通过时间隙处理
位和命令字来确认信息交换。
------------------------------------------*/
bit ds_read_bit(void)    //读一位
{
   bit dat;
   ds=0;         //单片机(微处理器)将总线拉低
_nop_();       //读时隙起始于微处理器将总线拉低至少1us
   ds=1;        //拉低总线后接着释放总线,让从机18b20能够接管总线,输出有效数据
   _nop_();
   _nop_();          //小延时一下,读取18b20上的数据 ,因为从ds18b20上输出的数据
//在读"时间隙"下降沿出现15us内有效
   dat=ds;           //主机读从机18b20输出的数据,这些数据在读时隙的下降沿出现//15us内有效 
   TempDelay(10);    //所有读"时间隙"必须60~120us,这里77us
   return(dat);       //返回有效数据
}
uchar ds_read_byte(void ) //读一字节
{

uchar value,i,j;
value=0;           //一定别忘了给初值
for(i=0;i<8;i++)
{
    j=ds_read_bit();
     value=(j<<7)|(value>>1);   //这一步的说明在一个word文档里面
}
return(value);        //返回一个字节的数据
}
void ds_write_byte(uchar dat) //写一个字节
{
uchar i;
bit onebit;        //一定不要忘了,onebit是一位
for(i=1;i<=8;i++) 
{
    onebit=dat&0x01;
    dat=dat>>1;
if(onebit)      //写 1
{
ds=0;
_nop_();    
      _nop_();      //看时序图,至少延时1us,才产生写"时间隙" 
ds=1;       //写时间隙开始后的15μs内允许数据线拉到高电平
     TempDelay(5); //所有写时间隙必须最少持续60us
}
else         //写 0
{
ds=0;
     TempDelay(8);    //主机要生成一个写0 时间隙,必须把数据线拉到低电平并保持至少60μs,这里64us
ds=1;
_nop_();
     _nop_();
}
}
}

/***************************************** 
主机(单片机)控制18B20完成温度转换要经过三个步骤:
每一次读写之前都要18B20进行复位操作,复位成功后发送
一条ROM指令,最后发送RAM指令,这样才能对DS18b20进行
预定的操作。
复位要求主CPU将数据线下拉500us,然后释放,当ds18B20
受到信号后等待16~60us,后发出60~240us的存在低脉冲,
主CPU收到此信号表示复位成功
******************************************/

/*----------------------------------------
进行温度转换:
先初始化
然后跳过ROM:跳过64位ROM地址,直接向ds18B20发温度转换命令,适合单片工作
发送温度转换命令
------------------------------------------*/

void tem_change()
{
ds_reset(); 
delay(1);              //约2ms
ds_write_byte(0xcc);
ds_write_byte(0x44);
}

/*----------------------------------------
获得温度:
------------------------------------------*/
uint get_temperature()
{
float wendu;
uchar a,b;
ds_reset();
delay(1);              //约2ms
ds_write_byte(0xcc);
ds_write_byte(0xbe);
a=ds_read_byte();
b=ds_read_byte();
temp=b;
temp<<=8;
temp=temp|a;
wendu=temp*0.0625;     //得到真实十进制温度值,因为DS18B20
                                    //可以精确到0.0625度,所以读回数据的最低位代表的是 //0.0625度

temp=wendu*10+0.5;       //放大十倍,这样做的目的将小数点后第一位
                                       //也转换为可显示数字,同时进行一个四舍五入操作。
return temp;
}
/*----------------------------------------
读ROM   
------------------------------------------*/
/*
void ds_read_rom()                  //这里没有用到
{
   uchar a,b;
   ds_reset();
   delay(30);
   ds_write_byte(0x33);
   a=ds_read_byte();
   b=ds_read_byte();
}
*/
void main()
{
   uint a;
   init_com();
   while(1)
   {
tem_change();          //12位转换时间最大为750ms
for(a=10;a>0;a--)
{
display( get_temperature());
}
   }
}

关键字:单片机  DS18B20  温度传感器 引用地址:单片机中使用DS18B20温度传感器C语言程序(参考1)

上一篇:单片机中使用DS18B20温度传感器C语言程序(参考2)
下一篇:MCS-51系列单片机的中断优先级

推荐阅读最新更新时间:2024-03-16 15:10

MCU 是如何从上电复位运行到 main 函数的?
前言 在笔者的上一篇文章中《中断服务子程序是如何被执行的》,详细阐述了中断响应以及执行的整个过程,其中涉及到关于中断向量表的相关知识,本篇文章再次以中断向量表为出发点阐述上电复位的整个过程。 复位的相关概念 复位就类似于我们的个人 PC 重启一样,又比 PC 的重启要简单一些。引起复位的原因也是多种多样,笔者在这里大致列出以下几种: 上电复位,也就是我们给我们的 MCU 通电后,其实也是一次复位的过程。 外部产生的手动复位信号,这个也比较常见,我们在平时学习所使用的开发板中就存在一个复位的按键,来实现手动的复位信号。 执行复位指令引发的复位 看门狗复位 。。。。。。 上述所示的复位虽然引起复位的原因各不相同,但是其复位的过
[单片机]
<font color='red'>MCU</font> 是如何从上电复位运行到 main 函数的?
单片机+DHT11+1602显示屏
单片机源程序如下: /********************************************************************************* 本实验仅限学习交流 AT89C52 12Mhz DHT11 LCD1602 接线如下 DHT11_DQ = P2^0 LCD1602_E= P2^7 LCD1602_RW=P2^5 LCD1602_RS=P2^6 D0—D7接到P0口 本实验没有用到DHT的校验位Check_Init,想使用校验位,定义一个变量t,t=HUMI_Buffer_Int+HUMI_Low_Int+TEM_Buffer_Int+TEM_
[单片机]
<font color='red'>单片机</font>+DHT11+1602显示屏
用51单片机中断控制LED灯亮灭
#include reg51.h //头文件 sbit LED=P2^0;//位定义LED灯 sbit k3=P3^2;//位定义按键 void delay()//延时函数 { unsigned char a,b; for(a=0;a 200;a++) for(b=0;b 200;b++); } void Int0()//使用中断所做的处理函数 { IT0=1;//外部中断0下降沿触发 EX0=1;//打开外部中断0 EA=1;//打开总中断 } void main()//主函数 { Int0();//调用函数 while(1); } void ledxs() interrupt
[单片机]
51单片机入门点亮LED
  一.keil新建单片机工程   1.单击”project”选项中的“new uvision project”,输入工程姓名。      2.挑选单片机类型,本系列以AT89C52单片机为政策,所以在Ateml中挑选AT89C52,右上角的方框不需求打勾,默许即可。右下角的”Description”便是对AT89C52单片机的描写,包括8位单片机,守时器和IO口数量,Flash容量等根柢信息。      点击OK往后,会出来一个对话框,意思是是不是复制主张代码到工程并增加文件,入门时期咱们的首要意图是了解单片机各个内置模块怎样运用,相应的C代码烧录后会有啥样的作用,这儿咱们选“否”就能够了。      3.建好工程往后,接
[单片机]
51<font color='red'>单片机</font>入门点亮LED
51单片机实现两片联级74HC595控制跑马灯
一、使用proteus绘制简单的电路图,用于后续仿真 二、编写程序 /******************************************************************************************************************** ---- @Project: LED-74HC595 ---- @File: main.c ---- @Edit: ZHQ ---- @Version: V1.0 ---- @CreationTime: 20200529 ---- @ModifiedTime: 20200529
[单片机]
51<font color='red'>单片机</font>实现两片联级74HC595控制跑马灯
单片机控制MGLS-12864接口研究
  随着传统的数字显示已经不能表达复杂的信息。而液晶显示器(LCD)则以其重量轻、电压低、功耗小、显示内容丰富等优点逐渐在单片机控制的智能仪器仪表、工业控制等领域得到广泛应用。   矿用隔爆馈电开关是煤矿井下配电系统的关键设备,其性能好坏直接影响煤矿井下的生产安全和生产效率。目前大部分矿井馈电开关保护器的显示通常是发光二极管或LED数字显示,并且大部分用灯来显示故障。   为了改进馈电开关的显示功能,利用单片机8031和液晶显示器设计了一种具有汉字显示功能的显示系统,该系统能够通过液晶显示器实时显示开关的工作参数(电压、电流等)、工作状态、供电线路的故障原因以及故障所发生的时间等,并且能对故障状态进行存储,大大方便了线路的维修
[单片机]
<font color='red'>单片机</font>控制MGLS-12864接口研究
单片机式语音播报伏特表
摘要: 本文将介绍一种基于单片机的语音播报伏特表,该系统可以对直流电压进行比较高精度的测量并用语音播报该值。同时,亦可以通过改变ADC中REF(—)和REF(+)的值实现对不同范围的直流电压的测量。 关键词: 单片机 语音播报 模数转换(ADC) 伏特表 引言 传统的伏特表在我们的日常生活及科学研究中起到了其独特的作用,但是在科学技术日新月异、集成芯片在日常生活中的应用越来越广泛的今天显得比较落伍:①它们的量程往往在出厂以前就限定好的,不能根据具体使用场合进行相应调整;②测量精度有限;③不能够将测量结果用语音播放出来。本文将介绍一种由单片机最小系统 、模-数转换电路 、语音电路 、LED显示电路 组成的单片机式语
[单片机]
基于51单片机的大气温度采集和记录系统
1 引 言 大气温度的测量具有重要意义, 一般是采用各种温度计进行人工测量, 并定时记录测量结果, 这种方法费时费工, 十分不便。本文提出了一种非常小巧的大气温度测量和记录装置, 能对范围为- 55 ℃ ~ + 125 ℃ 的温度进行定时自动测量, 同时记录测量结果, 精度最高可达0. 0625℃ 。 2 系统原理 系统结构如图1所示。系统主要由AT89C2051单片机和外围电路构成。其基本思想是由串行时钟芯片DS1306 提供时间基准, 由温度传感器DS18B20进行温度测量, 温度数据存储在DS2433中。AT89C2051单片机负责整个系统的协调控制。 AT89C2051是一款低电压, 高性能的CMOS 8位单片机, 片内含
[单片机]
基于51<font color='red'>单片机</font>的大气温度采集和记录系统
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved