下面是我写的温度转换函数,我不知道为什么,在液晶上面显示出来的不是当前的温度还是一个乱的温度。希望可以指点一下:
Get_Temp()
{
uchar TH,TL;
uint temperature;
float temp=0.1;
Init_DS18B20(); //初始化化18B20
WriteOneChar(0xcc); //跳过度序列号
WriteOneChar(0x44); //启动温度转换
delay_nms(100);
Init_DS18B20(); //初始化化18B20
WriteOneChar(0xcc); //跳过度序列号
WriteOneChar(0xbe); //读取温度寄存器
delay_nms(100);
TH = ReadOneChar(); //读取温度的高位
TL = ReadOneChar(); //读取温度的低位
temp = TH; //把高八位给中间变量
temperature = temperature<<8; //向左移动8位,为变成16位做准备
temperature = temperature|TL; //变成16的温度值 ,即是把两个字节变成一个int数据类型
if((temperature&0xf800)==0xf800) //判断温度是否为负值,因为18B20温度采集的高八位有5位是符号标志位,如果是负温度,这5位全为1
{ //所以就将得到我温度值和0xf800相与,取出前面的5位,看着五位是不是1;
mark=1; //如果是负值,标志位为1
temperature =~(temperature)+1; //如果是负数,温度取反再加一
}
else
{
mark=0; //如果是正数,那么标志位为0
}
temp=temperature*(0.0625)*10; //温度值扩大10倍,
return temp; //temp是整型
}
还有我在想是不是显示函数出了问题,下面是显示函数,是可以显示字符的,但是就是显示的温度是乱的。是不是我对温度的处理错了,可以帮忙看看。
显示函数:
void display_temp(uchar temp1)
{
uchar temp1;
uchar flagdat;
temp1=Get_temp();
dispdata[0]=temp1/1000+0x30;//取百位数
dispdata[1]=temp1%1000/100+0x30;//取十位数
dispdata[2]=temp1%100/10+0x30;//取个位数
dispdata[3]=temp1%10+0x30; //取小数
if(mark==0)
flagdat=0x20;//不显示负号
else
flagdat=0x2d; //显示负?
if(dispdata[0]==0x30)
{
dispdata[0]=0x20;//不显示0
if(dispdata[1]==0x30)
dispdata[1]=0x20;//不显示0
}
LCD_pos(0x05);// 从0x30位置开始显示
write_data('D');
write_data('S');
write_data('1');
write_data('8');
write_data('B');
write_data('2');
write_data('0');
delay_nms(30);
LCD_pos(0x42);//第二排第二个位置开始显示;
write_data('t');
write_data('e');
write_data('m');
write_data('p');
write_data(':');
delay_nms(30);
write_data(flagdat);
write_data(dispdata[0]);
write_data(dispdata[1]);
write_data(dispdata[2]);
write_data(0x2e);
// write_data(dispdata[3]);
write_data(0xdf);
write_data('C');
delay_nms(10);
}
#include<reg51.h>
#define uchar unsigned char
#define uint unsigned int
sbit DQ=P3^7;//ds18b20与单片机连接口
sbit RS=P2^6;
sbit RW=P2^5;
sbit EN=P2^7;
unsigned char code str1[]={"temperature: "};
unsigned char code str2[]={" °C "};
uchar data disdata[5];
uint tvalue;//温度值
uchar tflag;//温度正负标志
/*************************lcd1602程序**************************/
void delay1ms(unsigned int ms)//延时1毫秒(不够精确的)
{unsigned int i,j;
for(i=0;i<ms;i++)
for(j=0;j<100;j++);
}
void wr_com(unsigned char com)//写指令//
{ delay1ms(1);
RS=0;
RW=0;
EN=0;
P0=com;
delay1ms(1);
EN=1;
delay1ms(1);
EN=0;
}
void wr_dat(unsigned char dat)//写数据//
{ delay1ms(1);;
RS=1;
RW=0;
EN=0;
P0=dat;
delay1ms(1);
EN=1;
delay1ms(1);
EN=0;
}
void lcd_init()//初始化设置//
{delay1ms(15);
wr_com(0x38);delay1ms(5);
wr_com(0x08);delay1ms(5);
wr_com(0x01);delay1ms(5);
wr_com(0x06);delay1ms(5);
wr_com(0x0c);delay1ms(5);
}
void display(unsigned char *p)//显示//
{
while(*p!='\0')
{
wr_dat(*p);
p++;
delay1ms(1);
}
}
init_play()//初始化显示
{ lcd_init();
wr_com(0x80);
display(str1);
wr_com(0xc0);
display(str2);
}
/******************************ds1820程序***************************************/
void delay_18B20(unsigned int i)//延时1微秒
{
while(i--);
}
void ds1820rst()/*ds1820复位*/
{ unsigned char x=0;
DQ = 1; //DQ复位
delay_18B20(4); //延时
DQ = 0; //DQ拉低
delay_18B20(100); //精确延时大于480us
DQ = 1; //拉高
delay_18B20(40);
}
uchar ds1820rd()/*读数据*/
{ unsigned char i=0;
unsigned char dat = 0;
for (i=8;i>0;i--)
{ DQ = 0; //给脉冲信号
dat>>=1;
DQ = 1; //给脉冲信号
if(DQ)
dat|=0x80;
delay_18B20(10);
}
return(dat);
}
void ds1820wr(uchar wdata)/*写数据*/
{unsigned char i=0;
for (i=8; i>0; i--)
{ DQ = 0;
DQ = wdata&0x01;
delay_18B20(10);
DQ = 1;
wdata>>=1;
}
}
read_temp()/*读取温度值并转换*/
{uchar a,b;
ds1820rst();
ds1820wr(0xcc);//*跳过读序列号*/
ds1820wr(0x44);//*启动温度转换*/
ds1820rst();
ds1820wr(0xcc);//*跳过读序列号*/
ds1820wr(0xbe);//*读取温度*/
a=ds1820rd();
b=ds1820rd();
tvalue=b;
tvalue<<=8;
tvalue=tvalue|a;
if(tvalue<0x0fff)
tflag=0;
else
{tvalue=~tvalue+1;
tflag=1;
}
tvalue=tvalue*(0.625);//温度值扩大10倍,精确到1位小数
return(tvalue);
}
/*******************************************************************/
void ds1820disp()//温度值显示
{ uchar flagdat;
disdata[0]=tvalue/1000+0x30;//百位数
disdata[1]=tvalue%1000/100+0x30;//十位数
disdata[2]=tvalue%100/10+0x30;//个位数
disdata[3]=tvalue%10+0x30;//小数位
if(tflag==0)
flagdat=0x20;//正温度不显示符号
else
flagdat=0x2d;//负温度显示负号:-
if(disdata[0]==0x30)
{disdata[0]=0x20;//如果百位为0,不显示
if(disdata[1]==0x30)
{disdata[1]=0x20;//如果百位为0,十位为0也不显示
}
}
wr_com(0xc0);
wr_dat(flagdat);//显示符号位
wr_com(0xc1);
wr_dat(disdata[0]);//显示百位
wr_com(0xc2);
wr_dat(disdata[1]);//显示十位
wr_com(0xc3);
wr_dat(disdata[2]);//显示个位
wr_com(0xc4);
wr_dat(0x2e);//显示小数点
wr_com(0xc5);
wr_dat(disdata[3]);//显示小数位
}
/********************主程序***********************************/
void main()
{ init_play();//初始化显示
while(1)
{read_temp();//读取温度
ds1820disp();//显示
}
}
测试通过的程序,看看,你的有点乱
18B20程序无法读出温度,找不出原因...
这是延时时间的问题 include<reg52.h> include"ds18b20.h"define uchar unsigned char define uint unsigned int sbit DQ=P3^3;uchar temp_data[4];\/ 函数名: void delay_us(uint xus)功能: 延时时间为6us 调用函数:无 输入参数:xms,6us计数 输出参数:无 说明:总共延时时间为6us乘...
DS18B20 开发板上面显示温度 Proteus上面不显示温度是什么情况。按照开...
DS18B20对时序要求非常严格,如果读\/写操作时序与要求不符,就不能正确读出温度。同样的程序在开发板上可以读出温度,但仿真就读不出来。反之,也一样。程序都要做相应的调整,其实就是调延时的长短。如果仿真中单片机的时钟频率(注,并不是外部晶振频率,仿真中的晶振是无用的,是可以省掉的),与开发...
18b20 在 proteus 中为何 读不出温度数值
很有可能是主机发出温度转换命令(0x44)后,由于程序设定的延迟时间不够长,温度转换没有完成就开始读取数据了
DS18B20为什么读不出来温度?
uchar ds1820rd()\/*读数据*\/ { unsigned char i=0;
DS18B20温度显示的问题
85度是18b20的默认值,一般没有初始化成功会读出这个温度。另外18b20的转换时间很长,毫秒级的,这个要注意一下,最好采用判断DQ信号线的方法看温度是否转换完成。你的低温度可能跟这个有关。
DS18B20的一个程序里,为什么j用unsigned char的话就不能得到温度,下面...
DS18B20 对时序,要求很严格。读写脉冲的脉宽、间隔,都需要精确调试,它,才能正常工作。DS=0; \/\/write 0 j=8;while(j>0)j--;---延时 DS=1;延时部分的《 j--》,是 8 位数减一,还是 16 位数减一?你改变了数据类型,那么,延时时间,显然是不同的。延时时间不同,18B20,它就能...
ds18b20温度传感器故障
线路故障,没有采集到温度。DS18B20里面固化的温度就是85度,可以修改。
为什么我做的DS18B20数码管显示的温度不准确?
首先,要检查你的程序读出的DS18B20的温度值有没有问题,比如仿真中设定26°C,在程序中判断是不是26°C,若是,编一个简单的程序,让数码管输出1,这样就可以判定你的测温程序有没有问题。其次,最怀疑的地方就是你这个数码管的程序有问题,若你可以保证没有问题,可以排除这一块。最后,注意各个...
DS18B20总是显示 0摄氏度
lcdInit() 放外面while外面。显示函数放在读取温度函数的下面,启动温度转换后延时太久了吧,好像转换一次对应一次数据。另外可以软件测试下复位DS18B20函数能否通过。
ds18b20读取的温度始终是95.9度是咋回事呀。。
读取失败,或者说你发送或接受的数据格式错误。