首页 > 编程知识 正文

ds18b20温度转换详细(ds18b20高温下ff)

时间:2023-05-06 08:19:16 阅读:63886 作者:3873

使用LCD1602显示DS18B20温度传感器的温度。 DS18B20和液晶屏的原理不再赘述。 请大家自己找。 在网上可以找到很多。

使用Proteus模拟。 (如果需要模拟和Keil工程文件,请留言并查看回复) ) ) )。

所需设备: AT89C51、液晶屏1602和DS18B20。

电路图:

主函数代码: # includereg 52.h # included s18 b 20.h # include LCD 1602.hbi tflag 1s=0; //1s正时标记unsigned char T0RH=0; //T0重载值的高字节无符号字符t0rl=0; //T0重载值的低字节voidconfigtimer0(unsignedintms ); unsignedcharinttostring (unsigned char * str,int dat ); 扩展位启动18 b 20 (; 扩展字节18 b 20 temp (int * temp ); 扩展语音导入液晶屏1602 (; externvoidlcdshowstr (unsignedcharx,unsigned char y,unsigned char *str ); void main () {bit res; int temp; //读取的当前温度值int intT,decT; //温度值的整数和小数部分unsigned char len; 无符号char str [ 12 ]; EA=1; //总中断配置时间0 (10; //T0定时10msStart18B20 (; 启动ds18 b 20 init液晶屏1602 (; //初始化液晶液晶显示器(); while(1)每if(flag1s ) /秒更新温度) {flag1s=0; RES=get18b20temp(temp; //当前温度if(RES )//读取成功的情况下,更新当前温度显示({intT=temp 4; //分离温度值整数部分decT=temp0xF; //分离温度值的小数部分len=intTostring(str,intt ); //整数部分为字符串str[len ]='.'; //添加小数点dect=(dect*10 )/16;//二进制小数部分是一位十进制位str[len ]=decT '0';//用空格将十进制小数点以下的字符填充到ASCII字符while(Len6)//6个字符的长度({str[len ]=' '; }str[len]=' '; //添加字符串结束符LCD showstr (9,1,str ); //液晶画面上显示的LCD showstr (14,1,' c ' ); //c添加温度}else //读取失败时,出现错误信息{ LCD showstr (0,0,' error!' ); }Start18B20 (; //重新启动下一个转换}}}/*将整数转换为字符串,str-字符串指针,等待日期转换,返回值-字符串长度*/unsignedcharinttostring (unsigned char * str,int 无符号char buf [6]; if(dat0) /为负数时,首先取绝对值,在指针上输入负号({dat=-dat; *str ='-'; len; (do ) /是从下位变换到上位的十进制组buf[i ]=dat % 10; dat /=10; (while ) dat0; len =i; //i最后一个值为有效字符的个数while(I--0)//将数组值转换为ascii码并反向复制到接收指针({*str =buf[i] '0); }*str=' '; //添加字符串结束符返回len; //返回字符串长度(() ) T0、ms-T0定时时间)/voidconfigtimer0(unsignedintms ) {unsigned long tmp; //临时变量tmp=11059200/12; //计时器计数频率tmp=(tmp*ms )/1000; //计算所需计数值tmp=65536 - tmp; //计算计时器的重载值tmp=tmp 12; //由中断响应延迟导致的误差t0rh=(未指定通道) tmp8); //计时器的过载值划分成高字节t0rl=(未指定字符) tmp; TMOD=0xF0; //清零T0的控制位TMOD |=0x01; //配置T0为图案1TH0=T0RH; //t0重载值TL0=T0RL; ET0=1; //使能T0中断TR0=1; T0}/* T0启动中断服务函数,完成1秒计时器*/void InterruptTimer0() interrupt 1{ )

static unsigned char tmr1s = 0;TH0 = T0RH; //重新加载重载值TL0 = T0RL;tmr1s++;if (tmr1s >= 100) //定时 1s{tmr1s = 0;flag1s = 1;}} DS18B20.h代码: #include <reg52.h>#include <intrins.h>sbit IO_18B20 = P3^2; //DS18B20 通信引脚/* 软件延时函数,延时时间(t*10)us */void DelayX10us(unsigned char t){do {_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();} while (--t);}/* 复位总线,获取存在脉冲,以启动一次读写操作 */bit Get18B20Ack(){bit ack;EA = 0; //禁止总中断IO_18B20 = 0; //产生 500us 复位脉冲DelayX10us(50);IO_18B20 = 1;DelayX10us(6); //延时 60usack = IO_18B20; //读取存在脉冲while(!IO_18B20); //等待存在脉冲结束EA = 1; //重新使能总中断return ack;}/* 向 DS18B20 写入一个字节, dat-待写入字节 */void Write18B20(unsigned char dat){unsigned char mask;EA = 0; //禁止总中断for (mask=0x01; mask!=0; mask<<=1) //低位在先,依次移出 8 个 bit{IO_18B20 = 0; //产生 2us 低电平脉冲_nop_();_nop_();if ((mask&dat) == 0) //输出该 bit 值IO_18B20 = 0;elseIO_18B20 = 1;DelayX10us(6); //延时 60usIO_18B20 = 1; //拉高通信引脚}EA = 1; //重新使能总中断}/* 从 DS18B20 读取一个字节,返回值-读到的字节 */unsigned char Read18B20(){unsigned char dat;unsigned char mask;EA = 0; //禁止总中断for (mask=0x01; mask!=0; mask<<=1) //低位在先,依次采集 8 个 bit{IO_18B20 = 0; //产生 2us 低电平脉冲_nop_();_nop_();IO_18B20 = 1; //结束低电平脉冲,等待 18B20 输出数据_nop_(); //延时 2us_nop_();if (!IO_18B20) //读取通信引脚上的值dat &= ~mask;elsedat |= mask;DelayX10us(6); //再延时 60us}EA = 1; //重新使能总中断return dat;}/* 启动一次 18B20 温度转换,返回值-表示是否启动成功 */bit Start18B20(){bit ack;ack = Get18B20Ack(); //执行总线复位,并获取 18B20 应答if (ack == 0) //如 18B20 正确应答,则启动一次转换{Write18B20(0xCC); //跳过 ROM 操作Write18B20(0x44); //启动一次温度转换}return ~ack; //ack==0 表示操作成功,所以返回值对其取反}/* 读取 DS18B20 转换的温度值,返回值-表示是否读取成功 */bit Get18B20Temp(int *temp){bit ack;unsigned char LSB, MSB; //16bit 温度值的低字节和高字节ack = Get18B20Ack(); //执行总线复位,并获取 18B20 应答if (ack == 0) //如 18B20 正确应答,则读取温度值{Write18B20(0xCC); //跳过 ROM 操作Write18B20(0xBE); //发送读命令LSB = Read18B20(); //读温度值的低字节MSB = Read18B20(); //读温度值的高字节*temp = ((int)MSB << 8) + LSB; //合成为 16bit 整型数}return ~ack; //ack==0 表示操作应答,所以返回值为其取反值} LCD1602.h代码: #include <reg52.h>#define LCD1602_DB P0sbit LCD1602_RS = P2^0;sbit LCD1602_RW = P2^1;sbit LCD1602_E = P2^2;void InitLcd1602();void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);void delay(unsigned int i){while(i--);}void LcdStar(){unsigned char str[] = "ngdwbl";unsigned char tab[]="wendu = "; InitLcd1602();/* 初始化 1602 液晶 */LcdShowStr(2, 0, str);LcdShowStr(1, 1, tab);LcdShowStr(9, 1, "...");//默认初始化温度00LcdShowStr(14, 1, "C");//添加C温度}/* 等待液晶准备好 */void LcdWaitReady(){unsigned char sta;LCD1602_DB = 0xFF;LCD1602_RS = 0;LCD1602_RW = 1;do {LCD1602_E = 1;sta = LCD1602_DB; //读取状态字LCD1602_E = 0;} while (sta & 0x80); //bit7 等于 1 表示液晶正忙,重复检测直到其等于 0 为止}/* 向 LCD1602 液晶写入一字节命令, cmd-待写入命令值 */void LcdWriteCmd(unsigned char cmd){LcdWaitReady();LCD1602_RS = 0;LCD1602_RW = 0;LCD1602_DB = cmd;LCD1602_E = 1;LCD1602_E = 0;}/* 向 LCD1602 液晶写入一字节数据, dat-待写入数据值 */void LcdWriteDat(unsigned char dat){LcdWaitReady();LCD1602_RS = 1;LCD1602_RW = 0;LCD1602_DB = dat;LCD1602_E = 1;LCD1602_E = 0;}/* 设置显示 RAM 起始地址,亦即光标位置, (x,y)-对应屏幕上的字符坐标 */void LcdSetCursor(unsigned char x, unsigned char y){unsigned char addr;if (y == 0) //由输入的屏幕坐标计算显示 RAM 的地址addr = 0x00 + x; //第一行字符地址从 0x00 起始elseaddr = 0x40 + x; //第二行字符地址从 0x40 起始LcdWriteCmd(addr | 0x80); //设置 RAM 地址}/* 在液晶上显示字符串, (x,y)-对应屏幕上的起始坐标, str-字符串指针 */void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str){LcdSetCursor(x, y); //设置起始地址while (*str != '') //连续写入字符串数据,直到检测到结束符{LcdWriteDat(*str++); //先取 str 指向的数据,然后 str 自加 1}}/* 初始化 1602 液晶 */void InitLcd1602(){LcdWriteCmd(0x38); //16*2 显示, 5*7 点阵, 8 位数据接口LcdWriteCmd(0x0C); //显示器开,光标关闭LcdWriteCmd(0x06); //文字不动,地址自动+1LcdWriteCmd(0x01); //清屏} 效果截图:

小手留情,点个赞呗。

版权声明:该文观点仅代表作者本人。处理文章:请发送邮件至 三1五14八八95#扣扣.com 举报,一经查实,本站将立刻删除。