wc86110 发表于 2016-11-3 12:52 刚刚已经解决了,但我还是不知道哪儿出了问题。带线的ds18b20和以前正常用的不带线的时序不同,上电85度,出错误时候显示的也是85度,开始把读温度函数里的延时全部加了点,不管用,后来把写函数延时加了点就好了。感觉是给1820初始化的时候时序不匹配,写入不成功,所以读取就不成功导致显示错误的。 |
| 应该在硬件上去查找问题, |
|
图片没有上传成功,正常温度显示17.5,间隔跳到85.0或-85.0.正常温度会正常变化,错误显示一直是85或-85 |
|
程序如下:用开发板的程序改了三极管驱动的数码管的程序 //温度显示程序==LED显示,精度0.1摄氏度 //晶振:12M /****************************************** 跳线设置:默认 注意事项:ds18b20切勿插反,有爆炸烫伤的危险,方向是ds18b20的平面(有字的一面)朝旁边的三极管Q4 ***/ #include <reg52.h> #include <intrins.h> #define uchar unsigned char #define uint unsigned int sbit DQ = P3^0; sbit led3=P3^2; sbit led4=P3^3; sbit led1=P3^4; sbit led2=P3^5; //数据口define interface uint temp,r; //温度值 variable of temperature //不带小数点 unsigned char code table[] = {0x48,0xee,0x54,0x64,0xe2,0x61,0x41, 0xec,0x40,0x60}; //带小数点 unsigned char code table1[] = {0x08,0xae,0x14,0x24,0xa2,0x21,0x01,0xac,0x00,0x20}; /*************精确延时函数*****************/ void delay(unsigned char i) { while(--i); } /****************************************** 此延时函数针对的是12Mhz的晶振 delay(0):延时518us 误差:518-2*256=6 delay(1):延时7us (原帖写"5us"是错的) delay(10):延时25us 误差:25-20=5 delay(20):延时45us 误差:45-40=5 delay(100):延时205us 误差:205-200=5 delay(200):延时405us 误差:405-400=5 *******************************************/ void pdelay(unsigned char i) { uint x,y; for(x=i;x>0;x--) for(y=110;y>0;y--); } /*****************DS18B20******************/ void Init_Ds18b20(void) //DS18B20初始化send reset and initialization command { DQ = 1; //DQ复位,不要也可行。 delay(1); //稍做延时 DQ = 0; //单片机拉低总线 delay(250); //精确延时,维持至少480us DQ = 1; //释放总线,即拉高了总线 delay(100); //此处延时有足够,确保能让DS18B20发出存在脉冲。 } uchar Read_One_Byte() //读取一个字节的数据read a byte date //读数据时,数据以字节的最低有效位先从总线移出 { uchar i = 0; uchar dat = 0; for(i=8;i>0;i--) { DQ = 0; //将总线拉低,要在1us之后释放总线 //单片机要在此下降沿后的15us内读数据才会有效。 _nop_(); //至少维持了1us,表示读时序开始 dat >>= 1; //让从总线上读到的位数据,依次从高位移动到低位。 DQ = 1; //释放总线,此后DS18B20会控制总线,把数据传输到总线上 delay(1); //延时7us,此处参照推荐的读时序图,尽量把控制器采样时间放到读时序后的15us内的最后部分 if(DQ) //控制器进行采样 { dat |= 0x80; //若总线为1,即DQ为1,那就把dat的最高位置1;若为0,则不进行处理,保持为0 } delay(10); //此延时不能少,确保读时序的长度60us。 } return (dat); } void Write_One_Byte(uchar dat) { uchar i = 0; for(i=8;i>0;i--) { DQ = 0; //拉低总线 _nop_(); //至少维持了1us,表示写时序(包括写0时序或写1时序)开始 DQ = dat&0x01; //从字节的最低位开始传输 //指令dat的最低位赋予给总线,必须在拉低总线后的15us内, //因为15us后DS18B20会对总线采样。 delay(10); //必须让写时序持续至少60us DQ = 1; //写完后,必须释放总线, dat >>= 1; delay(1); } } uint Get_Tmp() //获取温度get the temperature { float tt; uchar a,b,c; Init_Ds18b20(); //初始化 Write_One_Byte(0xcc); //忽略ROM指令 Write_One_Byte(0x44); //温度转换指令 Init_Ds18b20(); //初始化 Write_One_Byte(0xcc); //忽略ROM指令 Write_One_Byte(0xbe); //读暂存器指令 a = Read_One_Byte(); //读取到的第一个字节为温度LSB b = Read_One_Byte(); c=b&0x80; if(c==0x80) { r=1; temp=b; temp<<=8; temp=temp|a; tt=~temp+1; tt=tt*0.0652; temp=tt*10+0.5; return temp; } else { r=0; //读取到的第一个字节为温度MSB temp = b; //先把高八位有效数据赋于temp temp <<= 8; //把以上8位数据从temp低八位移到高八位 temp = temp|a; //两字节合成一个整型变量 tt = temp*0.0625; //得到真实十进制温度值 //因为DS18B20可以精确到0.0625度 //所以读回数据的最低位代表的是0.0625度 temp = tt*10+0.5; //放大十倍 return temp; //这样做的目的将小数点后第一位也转换为可显示数字 } //同时进行一个四舍五入操作。 } /****************数码码动态显示函数**************/ void Display(uint temp) //显示程序 { uchar A1,A2,A3; A1 = temp/100; //百位 A2 = temp%100/10; //十位 A3 = temp%10; //个位 if(r==1) { led1=1; P1=0xf7; pdelay(5); led1=0; } else { led1=0; } P1=table[A1]; //显示百位 led2=1; pdelay(5); led2=0; P1 = table1[A2]; //显示十位,使用的是有小数点的数组(因为temp值扩大了10倍,虽然是十位,实际为个位) led3 =1; pdelay(5); led3=0; P1 = table[A3]; //显示个位 led4 = 1; pdelay(5); led4=0; } void main() { while(1) { Display(Get_Tmp()); } } 17.5为正常温度值,显示期间不定时跳到错误显示然后恢复。 |