基于51单片机及DS18B20温度传感器的数字温度计程序及详细注释
基于51单片机的18B20温度计C程序

本程序为本人用单片机80c51与18b20做实验时写的,其中有些是汉语拼音,便于咱们中国同胞们理解.仅供参考#include <reg52.h>#define uint unsigned int#define uchar unsigned charuchar tab[]="0123456789WD:C";sbit ds=P2^6;sbit rs=P3^4;sbit rw=P3^5;sbit en=P3^6;uchar s,g,shu;/***延时2tus***/yanshi(uchar t){while(--t){;}}/***延时tms** */delay(uint t){uchar a,b;for(a=t;a>0;a--)for(b=110;b>0;b--);}/***18B20初始化***/chushi(){ds=0;yanshi(250); //延时500usds=1;yanshi(2);}/***等待应答***/yingda(){while(ds);while(~ds);yanshi(2);}/***写一个字节***/w_dat(uchar dat){uchar i;for(i=0;i<8;i++){dat=dat>>1;ds=0;yanshi(8);ds=CY;yanshi(30);ds=1;}}/***读一个字节***/ uchar du_dat(){uchar i,dat;for(i=0;i<8;i++){ dat=dat>>1;ds=0;yanshi(1);ds=1;if(ds==1){dat=dat|0x80;}yanshi(30);}return dat;}/***启动温度转换***/ zhuanhuan(){chushi();yingda();delay(1);w_dat(0xcc);w_dat(0x44);}/***读取温度***/uchar du_wendu(){uchar a,b,dat;chushi();yingda();delay(1);w_dat(0xcc);w_dat(0xbe);a=du_dat(); //低位字节;b=du_dat(); //高位字节;dat=(b<<4)|(a>>4);return dat;}////////////////////////////// 1602液晶/////////////////////////////w_ml(uchar com){rs=0;P0=com;delay(5);en=1;delay(5);en=0;}chushihua(){rw=0;en=0;w_ml(0x38);w_ml(0x0c);w_ml(0x06);w_ml(0x01); }w_1602(uchar date){rs=1;P0=date;delay(5);en=1;delay(5);en=0;}xianshi(){w_ml(0x80);w_1602(tab[10]);w_1602(tab[11]);w_1602(tab[12]);w_1602(tab[s]);w_1602(tab[g]);w_1602(tab[13]);}/***************//** 主函数**//**************/main(){chushihua();while(1){delay(1);zhuanhuan();delay(10);shu=du_wendu();s=shu/10;g=shu%10;xianshi();}}。
DS18B20温度传感器详细注解及驱动程序解析

DS18B20独特的单线接口方式,DS18B20在与微处理器连接时仅需要一条线即可实现微处理器与DS18B20的双向通讯。
测温范围-55℃~+125℃,固有测温误差(注意,不是分辨率,这里之前是错误的)1℃。
支持多点组网功能,多个DS18B20可以并联在唯一的三线上,最多只能并联8个,实现多点测温,如果数量过多,会使供电电源电压过低,从而造成信号传输的不稳定。
工作电源:3.0~5.5V/DC(可以数据线寄生电源)在使用中不需要任何外围元件。
测量结果以9~12位数字量方式串行传送。
如果使用51单片机的话,那么中间那个引脚必须接上4.7K~10K的上拉电阻,否则,由于高电平不能正常输入/输出,要么通电后立即显示85℃,要么用几个月后温度在85℃与正常值上乱跳。
根据DS18B20的通讯协议,主机(单片机)控制DS18B20完成温度转换必须经过三个步骤:每一次读写之前都要对DS18B20进行复位操作,复位成功后发送一条ROM指令,最后发送RAM指令,这样才能对DS18B20进行预定的操作。
读ROM 33H 读DS1820温度传感器ROM中的编码(即64位地址)发出此命令之后,接着发出64 位ROM 编码,访问单总线上与该编码相符合ROM 55H对应的DS1820 使之作出响应,为下一步对该DS1820 的读写作准备。
用于确定挂接在同一总线上DS1820 的个数和识别64 位ROM 地址。
为搜索ROM FOH操作各器件作好准备。
跳过ROM CCH 忽略64 位ROM 地址,直接向DS1820 发温度变换命令。
告警搜索ECH 执行后只有温度超过设定值上限或下限的片子才做出响应。
RAM命令启动DS1820进行温度转换,12位转换时最长为750ms(9位为93.75m温度变换44Hs)。
结果存入内部第0、1字节RAM中。
1、复位操作复位要求主CPU将数据线下拉500微秒,然后释放,当DS18B20收到信号后等待16~60微秒左右,后发出60~240微秒的存在低脉冲,主CPU收到此信号表示复位成功。
DS18B20数字温度计使用与C51程序

1.DS18B20基本知识DS18B20数字温度计是DALLAS公司生产的1-Wire,即单总线器件,具有线路简单,体积小的特点。
因此用它来组成一个测温系统,具有线路简单,在一根通信线,可以挂很多这样的数字温度计,十分方便。
1、DS18B20产品的特点(1)、只要求一个端口即可实现通信。
(2)、在DS18B20中的每个器件上都有独一无二的序列号。
(3)、实际应用中不需要外部任何元器件即可实现测温。
(4)、测量温度范围在-55。
C到+125。
C之间。
(5)、数字温度计的分辨率用户可以从9位到12位选择。
(6)、内部有温度上、下限告警设置。
2、DS18B20的引脚介绍TO-92封装的DS18B20的引脚排列见图1,其引脚功能描述见表1。
(底视图)图1表1 DS18B20详细引脚功能描述3.DS18B20的使用方法由于DS18B20采用的是1-Wire总线协议方式,即在一根数据线实现数据的双向传输,而对AT89S51单片机来说,硬件上并不支持单总线协议,因此,我们必须采用软件的方法来模拟单总线的协议时序来完成对DS18B20芯片的访问。
由于DS18B20是在一根I/O线上读写数据,因此,对读写的数据位有着严格的时序要求。
DS18B20有严格的通信协议来保证各位数据传输的正确性和完整性。
该协议定义了几种信号的时序:初始化时序、读时序、写时序。
所有时序都是将主机作为主设备,单总线器件作为从设备。
而每一次命令和数据的传输都是从主机主动启动写时序开始,如果要求单总线器件回送数据,在进行写命令后,主机需启动读时序完成数据接收。
数据和命令的传输都是低位在先。
DS18B20的复位时序DS18B20的读时序对于DS18B20的读时序分为读0时序和读1时序两个过程。
对于DS18B20的读时序是从主机把单总线拉低之后,在15us之内就得释放单总线,以让DS18B20把数据传输到单总线上。
DS18B20在完成一个读时序过程,至少需要60us才能完成。
基于51单片机和DS18B20的数字温度计设计

题目:基于89C51和DS18B20的数字温度计设计一、设计要求数字式温度计要求测温范围为-55~125°C,精度误差在0.1°C以内,LED 数码管直读显示。
二、方案论证根据系统的设计要求,选择DS18B20作为本系统的温度传感器,选择单片机AT89C51为测控系统的核心来完成数据采集、处理、显示、报警等功能。
选用数字温度传感器DS18B20,省却了采样/保持电路、运放、数/模转换电路以及进行长距离传输时的串/并转换电路,简化了电路,缩短了系统的工作时间,降低了系统的硬件成本。
该系统的总体设计思路如下:温度传感器DS18B20把所测得的温度发送到AT89C51单片机上,经过51单片机处理,将把温度在显示电路上显示,本系统显示器用4位共阳LED数码管以动态扫描法实现。
检测范围-55摄氏度到125摄氏度。
按照系统设计功能的要求,确定系统由3个模块组成:主控制器、测温电路和显示电路。
数字温度计总体电路结构框图如图1所示。
图1 数字温度计总体电路结构框图三、系统硬件电路的设计温度计电路设计原理图如图2所示,控制器使用单片机AT89C51,温度传感器使用DS18B20,用4位共阳LED数码管实现温度显示。
图2 数字温度计设计电路原理图1、主控制器AT89C51是一种带4K字节闪烁可编程可擦除只读存储器的低电压,高性能CMOS8位微处理器。
该器件采用ATMEL高密度非易失存储器制造技术制造,与工业标准的MCS-51指令集和输出管脚相兼容。
由于将多功能8位CPU和闪烁存储器组合在单个芯片中,ATMEL的AT89C51是一种高效微控制器,为很多嵌入式控制系统提供了一种灵活性高且价廉的方案。
2、显示电路显示电路采用4位共阳LED数码管,从P0口输出段码,列扫描用P3.0~P3.3口来实现,列驱动用8550三极管。
3、温度传感器工作原理DS18B20温度传感器是美国DALLAS半导体公司最新推出的一种改进型智能温度传感器,与传统的热敏电阻等测温元件相比,它能直接读出被测温度,并且可根据实际要求通过简单的编程实现9~12位的数字值读数方式。
基于AT89C51利用DS18B20做温度计的完整程序

sbit RS=P2^6;
sbit RW=P2^5;
sbit EN=P2^7;
uchar code str1[]={"temperature: "};
uchar data disdata[5];
uint tvalue;//温度值
{
uchar flagdat;
disdata[0]=tvalue/1000+0x30; //百位数
disdata[1]=tvalue%1000/100+0x30;//十位数
disdata[2]=tvalue%100/10+0x30;//个位数
disdata[3]=tvalue%10+0x30;//小数位
tflag=0;
else
{
tvalue=~tvalue+1;
tflag=1;
}
tvalue=tvalue*(0.625);//温
度值扩大10倍,精确到1位小数
return(tvalue);
uchar tflag;//温度正负标志
/*************************lcd1602程序**************************/
void delay1ms(uchar ms)//延时1毫秒(不够精确的)
{
uchar i;
uchar j;
for(i=0;i<ms;i++)
}
/*******************************************************************
单片机DS18B20温度计(有程序)

;单片机DS18B20温度计C语言程序; 有程序#include<reg51.h>#include<intrins.h>#include <math.H> //要用到取绝对值函数abs()//通过DS18B20测试当前环境温度, 并通过数码管显示当前温度值, 目前显示范围: -55~ +125度sbit wela = P2^7; //数码管位选sbit dula = P2^6; //数码管段选sbit ds = P2^2;int tempValue;//0-F数码管的编码(共阳极)unsigned char code table[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e};//0-9数码管的编码(共阳极), 带小数点unsigned char code tableWidthDot[]={0x40, 0x79, 0x24, 0x30,0x19, 0x12, 0x02,0x78, 0x00, 0x10};//延时函数, 对于11.0592MHz时钟, 例i=10,则大概延时10ms.void delay(unsigned int i){unsigned int j;while(i--){for(j = 0; j < 125; j++);}}//初始化DS18B20//让DS18B20一段相对长时间低电平, 然后一段相对非常短时间高电平, 即可启动 void dsInit(){//对于11.0592MHz时钟, unsigned int型的i, 作一个i++操作的时间大于?us unsigned int i;ds = 0;i = 100; //拉低约800us, 符合协议要求的480us以上while(i>0) i--;ds = 1; //产生一个上升沿, 进入等待应答状态i = 4;while(i>0) i--;}void dsWait(){unsigned int i;while(ds);while(~ds); //检测到应答脉冲i = 4;while(i > 0) i--;}//向DS18B20读取一位数据//读一位, 让DS18B20一小周期低电平, 然后两小周期高电平,//之后DS18B20则会输出持续一段时间的一位数据bit readBit(){unsigned int i;bit b;ds = 0;i++; //延时约8us, 符合协议要求至少保持1usds = 1;i++; i++; //延时约16us, 符合协议要求的至少延时15us以上b = ds;i = 8;while(i>0) i--; //延时约64us, 符合读时隙不低于60us要求return b;}//读取一字节数据, 通过调用readBit()来实现unsigned char readByte(){unsigned int i;unsigned char j, dat;dat = 0;for(i=0; i<8; i++){j = readBit();//最先读出的是最低位数据dat = (j << 7) | (dat >> 1);}return dat;}//向DS18B20写入一字节数据void writeByte(unsigned char dat){unsigned int i;unsigned char j;bit b;for(j = 0; j < 8; j++){b = dat & 0x01;dat >>= 1;//写"1", 将DQ拉低15us后, 在15us~60us内将DQ拉高, 即完成写1if(b){ds = 0;i++; i++; //拉低约16us, 符号要求15~60us内ds = 1;i = 8; while(i>0) i--; //延时约64us, 符合写时隙不低于60us要求}else //写"0", 将DQ拉低60us~120usds = 0;i = 8; while(i>0) i--; //拉低约64us, 符号要求ds = 1;i++; i++; //整个写0时隙过程已经超过60us, 这里就不用像写1那样, 再延时64us 了}}//向DS18B20发送温度转换命令void sendChangeCmd(){dsInit(); //初始化DS18B20, 无论什么命令, 首先都要发起初始化dsWait(); //等待DS18B20应答delay(1); //延时1ms, 因为DS18B20会拉低DQ 60~240us作为应答信号writeByte(0xcc); //写入跳过序列号命令字Skip RomwriteByte(0x44); //写入温度转换命令字Convert T}//向DS18B20发送读取数据命令void sendReadCmd(){dsInit();dsWait();delay(1);writeByte(0xcc); //写入跳过序列号命令字Skip RomwriteByte(0xbe); //写入读取数据令字Read Scratchpad}//获取当前温度值int getTmpValue(){unsigned int tmpvalue;int value; //存放温度数值float t;unsigned char low, high;sendReadCmd();//连续读取两个字节数据low = readByte();high = readByte();//将高低两个字节合成一个整形变量//计算机中对于负数是利用补码来表示的//若是负值, 读取出来的数值是用补码表示的, 可直接赋值给int型的valuetmpvalue = high;tmpvalue <<= 8;tmpvalue |= low;value = tmpvalue;//使用DS18B20的默认分辨率12位, 精确度为0.0625度, 即读回数据的最低位代表0.0625度t = value * 0.0625;//将它放大100倍, 使显示时可显示小数点后两位, 并对小数点后第三进行4舍5入//如t=11.0625, 进行计数后, 得到value = 1106, 即11.06 度//如t=-11.0625, 进行计数后, 得到value = -1106, 即-11.06 度value = t * 100 + (value > 0 ? 0.5 : -0.5); //大于0加0.5, 小于0减0.5return value;}unsigned char const timeCount = 3; //动态扫描的时间间隔//显示当前温度值, 精确到小数点后一位//若先位选再段选, 由于IO口默认输出高电平, 所以当先位选会使数码管出现乱码void display(int v){unsigned char count;unsigned char datas[] = {0, 0, 0, 0, 0};unsigned int tmp = abs(v);datas[0] = tmp / 10000;datas[1] = tmp % 10000 / 1000;datas[2] = tmp % 1000 / 100;datas[3] = tmp % 100 / 10;datas[4] = tmp % 10;if(v < 0){//关位选, 去除对上一位的影响P0 = 0xff;wela = 0;//段选P0 = 0x40; //显示"-"号dula = 1; //打开锁存, 给它一个下降沿量dula = 0;//位选P0 = 0xfe;wela = 1; //打开锁存, 给它一个下降沿量wela = 0;delay(timeCount);}for(count = 0; count != 5; count++){//关位选, 去除对上一位的影响P0 = 0xff;wela = 1; //打开锁存, 给它一个下降沿量wela = 0;//段选if(count != 2){P0 = table[datas[count]]; //显示数字}else{P0 = tableWidthDot[datas[count]]; //显示带小数点数字}dula = 0;//位选P0 = _crol_(0xfd, count); //选择第(count + 1) 个数码管wela = 1; //打开锁存, 给它一个下降沿量wela = 0;delay(timeCount);}}void main(){unsigned char i;while(1){//启动温度转换sendChangeCmd();//显示5次for(i = 0; i < 40; i++){display(tempValue);}tempValue = getTmpValue();}以下是我编的程序,可用#include <reg52.h>#include <intrins.h>//-----------------------------------------------------------sbit DQ=P1^5;//-----------------------------------------------------------unsigned char number[10]={0X3F,0X06,0X5B,0X4F,0X66,0X6D,0X7D,0X07,0X7F,0X6F};//数字0~9unsigned char wei[8]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80}; //数码管位循环unsigned char Flag;unsigned char Templ,Temph;unsigned int temp;//-----------------------------------------------------------//函数声明//-----------------------------------------------------------void delay(unsigned char i); //延时程序//----------------------------------void Int18b20(void); //18b20初始化void Write18b20(unsigned char dat); //向18b20写一字节unsigned char Read18b20(void); //从18b20读一字节void Start18b20(void); //开始转换温度void Get18b20(void); //读出温度void chinT(void); //数据转换//----------------------------------void display(void); //显示程序//-----------------------------------------------------------//函数功能:延时//-----------------------------------------------------------/*************精确延时函数*****************/void delay(unsigned char i){while(--i);}/*此延时函数针对的是12Mhz的晶振delay(0); //延时518us 误差:518-2*256=6delay(1); //延时7us (原帖写"5us"是错的)delay(10); //延时25us 误差:25-20=5delay(20); //延时45us 误差:45-40=5delay(100); //延时205us 误差:205-200=5delay(200); //延时405us 误差:405-400=5*///-----------------------------------------------------------//DS18b20的相关程序//-----------------------------------------------------------//初始化//-----------------------------------------------------------void Int18b20(void){DQ=1;_nop_();_nop_();DQ=0; //拉低delay(100); //延时205usdelay(200); //延时405us //等待400~960微秒,最短为480us DQ=1;delay(1); //延时7usdelay(20); //延时45us //等待15~60微秒(等待回复)if(DQ==1) //判断初始化的情况是否成功{Flag=0; //复位失败}else{Flag=1;while(!DQ); //等待回复完成}delay(200); //延时405us //等待完成初始化}//-----------------------------------------------------------//写一字节//-----------------------------------------------------------void Write18b20(unsigned char dat){unsigned char i;for(i=0;i<8;i++){DQ=1;_nop_();DQ=0;delay(1); //延时7us //拉低后延时小于15usif(dat&0x01){DQ=1;}else{DQ=0;}dat=dat>>1;delay(20); //延时45usdelay(10); //延时25us //延时大于60usDQ=1;delay(1); //延时7us //延时大于1us}}//-----------------------------------------------------------//读一字节//-----------------------------------------------------------unsigned char Read18b20(void){unsigned char i,dat=0;for(i=0;i<8;i++){DQ=1;_nop_();DQ=0;delay(1); //延时7usdat=dat>>1;DQ=1;delay(1); //延时7us //确保在15us后60us前读数据if(DQ){dat|=0x80;}delay(20); //延时45us //确保读时续大于60us}return dat;}//-----------------------------------------------------------//开始转换温度//-----------------------------------------------------------void Start18b20(void){Int18b20();Write18b20(0xcc); //跳过ROM指令Write18b20(0x44); //温度转换指令}//-----------------------------------------------------------//读出温度//-----------------------------------------------------------void Get18b20(void){Int18b20();Write18b20(0xcc); //跳过ROM指令Write18b20(0xbe); //读暂存器指令Templ=Read18b20();Temph=Read18b20();}//-----------------------------------------------------------//数据转换//-----------------------------------------------------------void chinT(void){float Tt;temp=Temph; //先把高八位有效数据赋于temptemp=(temp<<8); //将数据从temp低八位移到高八位temp=temp|Templ; //将两字节合成一个整型变量Tt=temp*0.0625; //得到真实十进制温度值(因为DS18B20可以精确到0.0625度) temp=Tt*10+0.5; //放大十倍(将小数点后一位变成个位,个位变成十位,十位变成百位,并四舍五入)}//-----------------------------------------------------------//显示程序//-----------------------------------------------------------void display(void){unsigned int i;unsigned char A1,A2,A3;A1=temp/100; //百位(温度的十位)A2=temp%100/10; //十位(温度的个位)A3=temp%10; //个位(温度的小数点后一位)for(i=0;i<20;i++){P0=0x00;P2=0x00;P0=number[A1];P2=wei[0];delay(220);P0=0x00;P2=0x00;P0=number[A2];P2=wei[1];delay(220);P0=0x00;P2=0x00;P0=number[A3];P2=wei[2];delay(220);P0=0x00;P2=0x00;P0=0x80;P2=wei[1];delay(220);}}//-----------------------------------------------------------//----------------------------------------------------------- void main(void){while(1){Int18b20();if(Flag){Start18b20(); //开始转换温度Get18b20(); //得到温度chinT(); //数据转换display(); //显示}else P3=0x01;}}。
51单片机Ds18b20温度传感器程序

* 实验名 : 18B20温度显示试验* 实验说明 : 数码管显示温度值,并且将温度值通过串口发送到电脑上。
* 连接方式 : 见连接图temp.h#ifndef __TEMP_H_#define __TEMP_H_#include<reg51.h>//---重定义关键词---//#ifndef uchar#define uchar unsigned char#endif#ifndef uint#define uint unsigned int#endif//--定义使用的IO口--//sbit DSPORT=P3^7;//--声明全局函数--//void Delay1ms(uint );uchar Ds18b20Init();void Ds18b20WriteByte(uchar com);uchar Ds18b20ReadByte();void Ds18b20ChangTemp();void Ds18b20ReadTempCom();int Ds18b20ReadTemp();#endiftemp.c#include"temp.h"/******************************************************************************** 函数名: Delay1ms* 函数功能: 延时函数* 输入: 无* 输出: 无*******************************************************************************/void Delay1ms(uint y){uint x;for( ; y>0; y--){for(x=110; x>0; x--);}}/******************************************************************************** 函数名: Ds18b20Init* 函数功能: 初始化* 输入: 无* 输出: 初始化成功返回1,失败返回0*******************************************************************************/ uchar Ds18b20Init(){uchar i;DSPORT = 0;//将总线拉低480us~960usi = 70;while(i--);//延时642usDSPORT = 1;//然后拉高总线,如果DS18B20做出反应会将在15us~60us后总线拉低i = 0;while(DSPORT)//等待DS18B20拉低总线{Delay1ms(1);i++;if(i>5)//等待>5MS{return 0;//初始化失败}}return 1;//初始化成功}/******************************************************************************** 函数名: Ds18b20WriteByte* 函数功能: 向18B20写入一个字节* 输入: com* 输出: 无*******************************************************************************/void Ds18b20WriteByte(uchar dat){uint i, j;for(j=0; j<8; j++){DSPORT = 0;//每写入一位数据之前先把总线拉低1usi++;DSPORT = dat & 0x01; //然后写入一个数据,从最低位开始i=6;while(i--); //延时68us,持续时间最少60usDSPORT = 1;//然后释放总线,至少1us给总线恢复时间才能接着写入第二个数值dat >>= 1;}}/******************************************************************************** 函数名: Ds18b20ReadByte* 函数功能: 读取一个字节* 输入: com* 输出: 无*******************************************************************************/ uchar Ds18b20ReadByte(){uchar byte, bi;uint i, j;for(j=8; j>0; j--){DSPORT = 0;//先将总线拉低1usi++;DSPORT = 1;//然后释放总线i++;i++;//延时6us等待数据稳定bi = DSPORT;//读取数据,从最低位开始读取/*将byte左移一位,然后与上右移7位后的bi,注意移动之后移掉那位补0。
基于51单片机和DS18B20的数字温度计设计说明

基于51单片机和DS18B20的数字温度计设计说明
1.硬件设计:
-51单片机:选择合适的型号,如STC89C52或AT89C52等。
-DS18B20温度传感器:该传感器是一种数字温度传感器,具有单总线接口和高精度测量能力。
-接口电路:将51单片机和DS18B20传感器连接起来,要注意电平转换和信号线的阻抗匹配。
2.软件设计:
-初始化:在主函数中,首先对单片机进行初始化设置,包括时钟设置、串口配置等。
-DS18B20通信协议:使用单总线协议与DS18B20传感器进行通信,包括发送复位信号、读写数据等操作。
-温度测量:通过向DS18B20发送读取温度的命令,从传感器中读取温度值并保存。
-数据传输:将温度值转换为可显示的格式,如摄氏度或华氏度,并通过串口输出或LED显示。
3.程序流程:
-初始化单片机,设置时钟和串口参数。
-进入主循环,循环执行以下操作:
-发送复位信号,启动温度转换。
-等待转换完成,发送读取温度命令。
-读取温度值,并进行数据处理转换。
-输出温度值。
4.其他功能:
-可以添加LCD显示模块,将温度值显示在液晶屏上。
-可以添加按键输入模块,通过按键切换温度单位或进行其他操作。
需要注意的是,该设计只是一个简单的示例,实际应用中可能需要根据具体需求进行扩展和修改。
同时,在程序设计过程中,也要注意低功耗和数据稳定性等方面的考虑。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
基于51单片机及DS18B20温度传感器的数字温度计程序(详细注释)电路实物图如下图所示:C语言程序如下所示:/********************************************************************zicreate----------------------------- Copyright (C) --------------------------* 程序名; 基于DS18B20的测温系统* 功能:实时测量温度,超过上下限报警,报警温度可手动调整。
K1是用来* 进入上下限调节模式的,当按一下K1进入上限调节模式,再按一下进入下限* 调节模式。
在正常模式下,按一下K2进入查看上限温度模式,显示1s左右自动* 退出;按一下K3进入查看下限温度模式,显示1s左右自动退出;按一下K4消除* 按键音,再按一下启动按键音。
在调节上下限温度模式下,K2是实现加1功能,* K1是实现减1功能,K3是用来设定上下限温度正负的。
* 编程者:Jason* 编程时间:2009/10/2*********************************************************************/#include<AT89X52.h> //将AT89X52.h头文件包含到主程序#include<intrins.h> //将intrins.h头文件包含到主程序(调用其中的_nop_()空操作函数延时)#define uint unsigned int //变量类型宏定义,用uint表示无符号整形(16位)#define uchar unsigned char //变量类型宏定义,用uchar表示无符号字符型(8位)uchar max=0x00,min=0x00; //max是上限报警温度,min是下限报警温度bit s=0; //s是调整上下限温度时温度闪烁的标志位,s=0不显示200ms,s=1显示1s左右bit s1=0; //s1标志位用于上下限查看时的显示void display1(uint z); //声明display1()函数#include"ds18b20.h" //将ds18b20.h头文件包含到主程序#include"keyscan.h" //将keyscan.h头文件包含到主程序#include"display.h" //将display.h头文件包含到主程序/***********************主函数************************/void main(){beer=1; //关闭蜂鸣器led=1; //关闭LED灯timer1_init(0); //初始化定时器1(未启动定时器1)get_temperature(1); //首次启动DS18B20获取温度(DS18B20上点后自动将EEPROM中的上下限温度复制到TH和TL寄存器)while(1) //主循环{keyscan(); //按键扫面函数get_temperature(0); //获取温度函数keyscan(); //按键扫面函数display(temp,temp_d*0.625);//显示函数alarm(); //报警函数keyscan(); //按键扫面函数}}/********************************************************************* 程序名; __ds18b20_h__* 功能:DS18B20的c51编程头文件* 编程者:ZPZ* 编程时间:2009/10/2* 说明:用到的全局变量是:无符号字符型变量temp(测得的温度整数部分),temp_d* (测得的温度小数部分),标志位f(测量温度的标志位‘0’表示“正温度”‘1’表* 示“负温度”),标志位f_max(上限温度的标志位‘0’表示“正温度”、‘1’表* 示“负温度”),标志位f_min(下限温度的标志位‘0’表示“正温度”、‘1’表* 示“负温度”),标志位w(报警标志位‘1’启动报警‘0’关闭报警)。
*********************************************************************/#ifndef __ds18b20_h__ //定义头文件#define __ds18b20_h__#define uint unsigned int //变量类型宏定义,用uint表示无符号整形(16位)#define uchar unsigned char //变量类型宏定义,用uchar表示无符号字符型(8位)sbit DQ= P2^3; //可位寻址变量定义,用DQ表示P2.3口sbit beer=P1^0; //用beer表示P1.0sbit led=P1^1; //用led表示P1.1uchar temp=0; //测量温度的整数部分uchar temp_d=0; //测量温度的小数部bit f=0; //测量温度的标志位,0’表示“正温度”‘1’表示“负温度”)bit f_max=0; //上限温度的标志位‘0’表示“正温度”‘1’表示“负温度”)bit f_min=0; //下限温度的标志位‘0’表示“正温度”、‘1’表示“负温度”)bit w=0; //报警标志位‘1’启动报警‘0’关闭报警)/*****************************延时子函数******************************/void ds18b20_delayus(uint t) //延时几μs{ while(t--);}void ds18b20_delayms(uint t) //延时1ms左右{ uint i,j; for(i=t;i>0;i--) for(j=120;j>0;j--);}/**************************ds18b20初始化函数*************************/void ds18b20_init() // DS18B20初始化{DQ=1; //拉高数据线DQ=0; //控制器向DS18B20发低电平脉冲ds18b20_delayus(30); //延时480μs左右DQ=1; //控制器拉高总线,while(DQ); //等待DS18B20拉低总线ds18b20_delayus(20); //延时,等待上拉电阻拉高总线DQ=1; //拉高数据线,准备数据传输;}/***************************ds18b20字节读函数************************/uchar ds18b20_read() //DS18B20 字节读取{uchar i; //定义一个局部变量i(局部变量只在本函数中有效)uchar d = 0; //定义一个局部变量dDQ = 1; //准备读;for(i=8;i>0;i--) //一位一位的读,循环8次{d >>= 1; //d左移一位,低位先发;DQ = 0;_nop_();_nop_();_nop_();DQ = 1; //必须写1,否则读出来的将是不预期的数据;if(DQ) //在12us处读取数据,送给d的最高位d |= 0x80;ds18b20_delayus(10);} return d; //返回读取的值}/*************************ds18b20字节写函数**************************/void ds18b20_write(uchar d) // ds18b20字节写{uchar i;for(i=8;i>0;i--) //一位一位的写{ DQ=0;_nop_(); _nop_();_nop_(); DQ=d&0x01; //写数据ds18b20_delayus(5); DQ=1; d >>= 1;}}/***************************获取温度函数****************************/void get_temperature(bit f){uchar a=0,b=0,c=0,d=0;uint i;ds18b20_init(); //DS18B20初始化ds18b20_write(0xcc);//向DS18B20发跳过读ROM命令ds18b20_write(0x44);//写启动DS18B20进行温度转换命令,转换结果存入内部RAMif(f==1){ //首次启动DS18B20进行温度转换需要500ms,若转换时间不够就出错,读出的是85度的错误值。
display1(1); //用开机动画耗时}elseds18b20_delayms(1);ds18b20_init(); //DS18B20初始化ds18b20_write(0xcc); //向DS18B20发跳过读ROM命令ds18b20_write(0xbe); //写读内部RAM中9字节的内容命令a=ds18b20_read(); //读内部RAM (LSB)b=ds18b20_read(); //读内部RAM (MSB)if(f==1) //局部位变量f=1时读上下线报警温度{max=ds18b20_read(); //读内部RAM (TH)min=ds18b20_read(); //读内部RAM (Tl)}if((max&0x80)==0x80) //若读取的上限温度的最高位(符号位)为‘1’表明是负温度{f_max=1;max=(max-0x80);} //将上限温度符号标志位置‘1’表示负温度,将上限温度装换成无符号数。
if((min&0x80)==0x80)//若读取的下限温度的最高位(符号位)为‘1’表明是负温度{f_min=1;min=(min-0x80);}//将下限温度符号标志位置‘1’表示负温度,将下限温度装换成无符号数。
i=b;i>>=4;if (i==0){f=0; //i为0,表示读取的温度是正温度,设立正温度标记temp=((a>>4)|(b<<4)); //整数部分a=(a&0x0f);temp_d=a; //小数部分}else{f=1; //i为1,表示读取的温度是负温度,设立负温度标记a=~a+1; //负数的小数部分取反加1b=~b; //负数的整数部分取反temp=((a>>4)|(b<<4)); //整数部分a=(a&0x0f); //小数部分temp_d=a;}}/*************************存储极限温度函数***************************/void store_t(){if(f_max==1) //若上限温度为负,将上限温度转换成有符号数(最高1是负,0是正)max=max+0x80;if(f_min==1) //若下限温度为负,将上限温度转换成有符号数min=min+0x80;ds18b20_init(); //DS18B20初始化ds18b20_write(0xcc); //向DS18B20发跳过读ROM命令ds18b20_write(0x4e); //向DS18B20发写字节至暂存器2和3(TH和TL)命令ds18b20_write(max); //向暂存器TH(上限温度暂存器)写温度ds18b20_write(min); //向暂存器TL(下限温度暂存器)写温度ds18b20_write(0xff); //向配置寄存器写命令,进行温度值分辨率设置ds18b20_init(); //DS18B20初始化ds18b20_write(0xcc); //向DS18B20发跳过读ROM命令ds18b20_write(0x48); //向DS18B20发将RAM中2、3字节的内容写入EEPROM} //DS18B20上电后会自动将EEPROM中的上下限温度拷贝到TH、TL暂存器/**************************温度超限报警函数*************************/void alarm(){ //若上限值是正值if(f_max==0){if(f_min==0) //若下限值是正值{if(f==0) //若测量值是正值{if((temp+temp_d*0.0625)<=min||(temp+temp_d*0.0625)>=max){w=1;TR1=1;} //当测量值小于最小值或大于最大值时报警if((temp+temp_d*0.0625)<max&&(temp+temp_d*0.0625)>min){w=0;} //当测量值大于最小值且小于最大值时不报警} if(f==1){w=1;TR1=1;} //若测量值是负值时报警}if(f_min==1) //若下限值是负值{ if(f==0) //若测量值是正值{if((temp+temp_d*0.0625)>=max)//当测量值大于最大值时报警{w=1;TR1=1;}if((temp+temp_d*0.0625)<max )//当测量值小于最大值时不报警{w=0;}}if(f==1) //若测量值是负值{ if((temp+temp_d*0.0625)>=min)//当测量值大于最小值时报警{w=1;TR1=1;}if((temp+temp_d*0.0625)<min)//当测量值小于最小值时不报警{w=0;}}}}if(f_max==1) //若下限值是负值{ if(f_min==1) //若下限值是负值{ if(f==1) //若测量值是负值{if((temp+temp_d*0.0625)<=max||(temp+temp_d*0.0625)>=min){w=1;TR1=1;} //当测量值小于最大值或大于最小值时报警if((temp+temp_d*0.0625)<min&&(temp+temp_d*0.0625)>max){w=0;} //当测量值小于最小值且大于最大值时不报警}if(f==0){w=1;TR1=1;} //若测量值是正值时报警}}}#endif/*********************************************************************** 程序名; __keyscan_H__* 功能:ds18b20键盘头文件,通过键盘设定设定上下限报警温度* 编程者:ZPZ* 编程时间:2009/10/2**********************************************************************/#ifndef __keyscan_H__ //定义头文件#define __keyscan_H__sbit key1=P2^2; //可位寻址变量定义,用key1表示P2.2口sbit key2=P2^1; //用key2表示P2.1口sbit key3=P2^0; //用key3表示P2.0口sbit key4=P3^3; //用key4表示P3.3口uchar i=0; //定义全局变量i用于不同功能模式的选择,‘0’正常模式,‘1’上限调节模式,‘2’下限调节模式uchar a=0; //定义全局变量a用于不同模式下数码管显示的选择bit k4=0; //K4按键双功能选择位,k4=0时K4按键选择消按键音的功能,k4=1时K4按键选择正负温度设定功能bit v=0; //K2、K3按键双功能选择位,v=0时选择上下限查看功能,v=1时选择上下限温度加减功能bit v1=0; //v1=1时定时1250ms时间到自动关闭报警上下限查看功能bit v2=0; //消按键音功能调整位,为‘0’时开按键音,为‘1’时关按键音/***************************读键盘延时子函数**************************/void keyscan_delay(uint z) //延时1ms左右{uint i,j;for(i=z;i>0;i--)for(j=120;j>0;j--);}/****************************温度调节函数******************************/int temp_change(int count,bit f) //上下限温度调整{if(key2==0) //判断K2是否按下{ if(v2==0)beer=0; //v2=0开按键音,否则消按键音keyscan_delay(10); //延时10msif(key2==0) //再次判断K2是否按下(实现按按键时消抖){beer=1; //K2按下关按键音if(f==0) //若温度为正{count++; //每按一下K2温度上调1if(a==1){if(count>125) count=125;}//当温度值大于125时不上调if(a==2){if(count>125) count=125;}}if(f!=0) //若温度为负{count++; //每按一下K2温度下调1if(a==1){if(count>55) count=55;}//当温度值小于-55时不再下调if(a==2){if(count>55) count=55;}}}while(key2==0); keyscan_delay(10); //K2松开按键时消抖}if(key3==0){if(v2==0)beer=0; keyscan_delay(10);if(key3==0) //K3按按键时消抖{beer=1;count--; //每按一下K3温度为正时下调1,为负时上调1if(a==1){if(count<0) count=0;}//当温度值达到0时不再调if(a==2){if(count<0) count=0;}}while(key3==0); keyscan_delay(10); //K3松开按键时消抖}return count;}/*****************************读键盘函数******************************/void keyscan(){ if(key1==0){if(v2==0) beer=0; keyscan_delay(10);if(key1==0) //K1按按键时消抖{beer=1;TR1=1;//开定时器1,通过s标志位的变化,实现在上下限温度调整时温度显示时闪烁的功能k4=1;//在上下温度调节功能模式下选择K4的调整上下限温度正负的功能v=1; //在上下温度调节功能模式下选择K2、K3的温度加减功能i++; //K1按一下i加1,i=‘0’进入正常模式,i=‘1’进入调上限模式,i=‘2’进入调下限模式if(i>2) //K1按下三次后退出调节模式{i=0; //进入正常模式TR1=0; //关定时器1k4=0; //在正常模式下选择K4的消按键音功能v=0; //在正常模式下选择K2、K3的查看上下限报警温度功能store_t(); //存储调整后的上下限报警温度}switch(i) //显示选择{case 0:a=0;break; //a=0选择显示测得的温度case 1:a=1;break; //a=1选择显示上限温度case 2:a=2;break; //a=2选择显示下限温度default:break;}}while(key1==0); //K1松按键时消抖keyscan_delay(10);}if(a==1&&v==1) //a=1选择显示上限温度且v=1时选择上下限温度加功能{led=0;max=temp_change(max,f_max);}//显示上限温度else if(a==2&&v==1) //a=2选择显示下限温度且v=1时选择上下限温度减功能{led=1;min=temp_change(min,f_min);}else;if(k4==1) //k4=1时K4按键选择正负温度设定功能{if(key4==0){if(v2==0)beer=0; keyscan_delay(5);if(key4==0){ beer=1;if(a==1){if(max>55) f_max=0;else f_max=~f_max;}//当温度大于55度时,只能设定为正温度if(a==2){if(min>55) f_max=0;else f_min=~f_min;}//当温度大于55度时,只能设定为正温度}while(key4==0); keyscan_delay(10);}}if(v==0) //v=0时选择上下限查看功能{if(key2==0){if(v2==0)beer=0; keyscan_delay(10);if(key2==0){beer=1;a=1; //选择上限显示TR1=1; //开定时器1开始定时一分钟左右s1=1; //上限显示不闪烁,显示一分钟左右自动退出}while(key2==0); keyscan_delay(10);}if(key3==0){if(v2==0)beer=0; keyscan_delay(10);if(key3==0){beer=1;a=2; //选择下限显示TR1=1; //开定时器1开始定时1ss1=1; //下限显示不闪烁,显示1s自动退出}while(key3==0); keyscan_delay(10);}if(v1==1) //v1=1时定时1s时间到自动关闭报警上下限查看功能{a=0;v1=0;TR1=0;} //a=0显示实测温度,v1清零,关定时器1if(k4==0) //k4=0时K4按键选择消按键音的功能{if(key4==0){if(v2==0)beer=0; keyscan_delay(10);if(key4==0){beer=1; v2=~v2; //为‘0’时开按键音,为‘1’时关按键音}while(key4==0);keyscan_delay(10);}}}}#endif/*********************************************************************** 程序名; __ds18b20_display_H__* 功能:ds18b20数码管动态显示头文件,通过定时器0延时实现数码管动态显示* 编程者:ZPZ* 编程时间:2009/10/2**********************************************************************/#ifndef __ds18b20_display_H__ //定义头文件#define __ds18b20_display_H__#define uint unsigned int //变量类型宏定义,用uint表示无符号整形(16位)#define uchar unsigned char //变量类型宏定义,用uchar表示无符号字符型(8位)sbit wei1=P2^4; //可位寻址变量定义,用wei1表示P2.4口sbit wei2=P2^5; //用wei2表示P2.5口sbit wei3=P2^6; //用wei3表示P2.6口sbit wei4=P2^7; //用wei4表示P2.7口uchar num=0; //定义num为全局无符号字符型变量,赋初值为‘0’uchar code temperature1[]={ 0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f}; //定义显示码表0~9uchar code temperature2[]={ 0xbf,0x86,0xdb,0xcf,0xe6,0xed,0xfd,0x87,0xff,0xef}; //带小数点的0.~9.uchar code temperature3[]={ 0x00,0x80,0x40,0x76,0x38}; //依次是‘不显示’‘.’‘-’‘H’‘L’/*****************************延时子函数******************************/void display_delay(uint t) //延时1ms左右{uint i,j;for(i=t;i>0;i--)for(j=120;j>0;j--);}/**************************定时器1初始化函数***************************/void timer1_init(bit t){TMOD=0x10; //设定定时器1工作在方式1,最大定时65.53ms TH0=0x3c; //定时器赋初值,定时50msTL0=0xb0;EA=1; //开总中断ET1=1; //开定时器1中断TR1=t; // 局部变量t为1启动定时器1,为0关闭定时器1 }/**************************定时器1中断函数*****************************/void timer1() interrupt 3{TH0=0x3c; //重新赋初值,定时50msTL0=0xb0;num++; //每进入一次定时器中断num加1(每50ms加1一次)if(num<5){s=1;if(w==1){beer=1;led=1;}else{beer=1;led=1;}}Else //进入4次中断,定时200ms时若报警标志位w为‘1’则启动报警,不为‘1’不启动//实现间歇性报警功能{s=0;if(w==1){beer=0;led=0;}else{beer=1;led=1;}}if(num>20) //进入20次中断,定时1s{num=0; //num归0,重新定开始定时1ss1=0; //定时1s时间到时自动关闭报警上下限显示功能v1=1; //定时1s时间到时自动关闭报警上下限查看功能}}/*********************调整报警上下限显示选择函数**********************/void selsct_1(uchar f,uchar k) //消除百位的0显示,及正负温度的显示选择{if(f==0) //若为正温度,百位为0则不显示百位,不为0则显示{if(k/100==0) P0=temperature3[0]; else P0=temperature1[k/100];}if(f==1) //若为负温度,若十位为0,百位不显示,否则百位显示‘-’{if(k%100/10==0) P0=temperature3[0]; else P0=temperature3[2];}}void selsct_2(bit f,uchar k) //消除十位的0显示,及正负温度的显示选择{if(f==0) //若为正温度,百位十位均为0则不显示十位,否则显示十位{ if((k/100==0)&&(k%100/10==0)) P0=temperature3[0];else P0=temperature1[k%100/10];}if(f==1) //若为负温度,若十位为0,十位不显示,否则十位显示‘-’{ if(k%100/10==0) P0=temperature3[2];else P0=temperature1[k%100/10];}}/****************************主显示函数********************************/void display(uchar t,uchar t_d) //用于实测温度、上限温度的显示{uchar i;for(i=0;i<4;i++) //依次从左至右选通数码管显示,实现动态显示{switch(i){case 0: //选通第一个数码管if(a==0){selsct_1(f,t);} //若a=0则在第一个数码管上显示测量温度的百位或‘-’if(a==1){P0=temperature3[3]; //若a=1则在第一个数码管上显示‘H’}if(a==2){P0=temperature3[4]; //若a=2则在第一个数码管上显示‘L’}wei2=0; //关第二个数码管wei3=0; //关第三个数码管wei4=0; //关第四个数码管wei1=1; //开第一个数码管break;case 1: //选通第二个数码管if(a==0){selsct_2(f,t);} //若a=0则在第二个数码管上显示测量温度的十位或‘-’if(a==1) //若a=1则在第二个数码管上显示上限报警温度的百位或‘-’{if(s==0) selsct_1(f_max,max);//若s=0则显示第二个数码管,否则不显示else P0=temperature3[0]; //通过s标志位的变化实现调节上下限报警温度时数码管的闪烁if(s1==1) selsct_1(f_max,max);//若s1=1则显示第二个数码管(s1标志位用于上下限查看时的显示)}if(a==2) //若a=2则在第二个数码管上显示下限报警温度的百位或‘-’{if(s==0) selsct_1(f_min,min);else P0=temperature3[0];if(s1==1) selsct_1(f_min,min);}wei1=0; wei3=0; wei4=0; wei2=1; break;case 2: //选通第三个数码管if(a==0){P0=temperature2[t%10];}//若a=0则在第三个数码管上显示测量温度的个位if(a==1) //若a=1则在第三个数码管上显示上限报警温度的十位或‘-’{if(s==0) selsct_2(f_max,max);//若s=0则显示第三个数码管,否则不显示else P0=temperature3[0];if(s1==1) selsct_2(f_max,max);//若s1=1则显示第三个数码管}if(a==2) //若a=2则在第三个数码管上显示下限报警温度的十位或‘-’{if(s==0) selsct_2(f_min,min);else P0=temperature3[0];if(s1==1) selsct_2(f_min,min);}wei1=0; wei2=0; wei4=0; wei3=1; break;case 3: //选通第四个数码管if(a==0){P0=temperature1[t_d];}//若a=0则在第四个数码管上显示测量温度的小数位if(a==1) //若a=1则在第四个数码管上显示上限报警温度的个位{if(s==0) P0=temperature1[max%10];//若s=0则显示第四个数码管,否则不显示else P0=temperature3[0];if(s1==1) P0=temperature1[max%10];//若s1=1则显示第四个数码管}if(a==2) //若a=2则在第四个数码管上显示下限报警温度的个位{if(s==0) P0=temperature1[min%10];else P0=temperature3[0];if(s1==1) P0=temperature1[min%10];}wei1=0; wei2=0; wei3=0; wei4=1; break;}display_delay(3); //每个数码管显示3ms左右}}/****************************开机显示函数******************************/ void display1(uint z) //用于开机动画的显示{uchar i,j;bit f=0;for(i=0;i<z;i++) //‘z’是显示遍数的设定{for(j=0;j<4;j++) //依次从左至右显示‘-’{switch(j){case 0:P0=temperature3[2];//第一个数码管显示wei2=0; wei3=0; wei4=0; wei1=1; break;case 1:P0=temperature3[2];//第二个数码管显示wei1=0; wei3=0; wei4=0; wei2=1; break;case 2:P0=temperature3[2];//第三个数码管显示wei1=0; wei2=0; wei4=0; wei3=1; break;case 3:P0=temperature3[2];//第四个数码管显示wei1=0; wei2=0; wei3=0; wei4=1; break;}display_delay(200); //每个数码管显示200ms左右}}}#endif。