ds18b20测温初学,测试失败啦麻烦大家给我看看

2020-02-05 09:20发布

本帖最后由 yesno 于 2012-5-19 15:29 编辑

#include<reg52.h>
#include<intrins.h>
#define uchar unsigned char  
#define uint unsigned int
sbit dula=P2^6;  
sbit wela=P2^7;
sbit DQ=P2^2;
uint temp;
uchar code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};          //0-9数字
void delay (uint x)                 //延时函数        
{        
uchar i;
        while(x--)
        for(i=0; i<120; i++);
        }
void delay1(uint a)                 //延时函数
{   
while(--a);
}
void disp(uchar num)  //显示函数
{
  uchar shi,ge;   
shi=num/10;   
ge=num%10;  
dula=1;  
P0=table[shi];
  dula=0;  
P0=0xff;   
wela=1;
  P0=0xfe;  
wela=0;   
delay(5);   
dula=1;  
P0=table[ge];  
dula=0;   
P0=0xff;  
wela=1;  
P0=0xfd;   
wela=0;   
delay(5);
}          
void init_18b20() //  初始化
{
DQ=1;
delay1(8);
DQ=0;
delay1(90);
DQ=1;
_nop_();
_nop_();
delay1(100);
DQ=1;
}        
void write(uchar dat)          //写字节
{
uchar i;        
for(i=0; i<8; i++)        
        {
                DQ=0;        
        DQ=dat&0x01;        
        delay1(5);                
        DQ=1;        
        dat>>=1;                
        }                
      delay1(4);
}  
uchar read()                   //读字节
{
uchar i, dat=0;
DQ=1;  
_nop_();  
for(i=0; i<8; i++)
{
   DQ=0;  
   _nop_();
   _nop_();
    dat>>=1;
    DQ=1;
    _nop_();
    _nop_();  
    if(DQ)
    dat|=0x80;
    delay1(30);
    DQ=1;
  }  
return dat;
}
uchar read_tu()                          //  温度转换
{
uchar a,b;
init_18b20();
delay1(100);
write(0xcc);
write(0x44);
init_18b20();
write(0xcc);
write(0xbe);
a=read();
b=read();
b<<=4;
b+=(a&0xf0)>>4;
return b;
}   
void main()                                                 //主函数
{
delay(30);
        while(1)
        {
        temp=read_tu;
        disp(temp);        
         }
}


这个初学仔细检查啦视乎没错,可是下载到板子里面就不行啦。其他的情况也没什么就是温度没反应,数码管就是显示70,任凭我怎么触摸传感器都是没反应既不升温也不降温。



友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
23条回答
yesno
1楼-- · 2020-02-06 20:55
BXAK 发表于 2012-5-19 15:24
编译环境是用keil还是TKStudio,晶振多少,懒得新建工程,你把这个项目的文件夹压缩成rar或者zip,然后上 ...

rar在上面啊
yesno
2楼-- · 2020-02-07 00:54
BXAK 发表于 2012-5-19 15:24
编译环境是用keil还是TKStudio,晶振多少,懒得新建工程,你把这个项目的文件夹压缩成rar或者zip,然后上 ...

测试的怎么样啦
無智
3楼-- · 2020-02-07 02:02
yesno 发表于 2012-5-19 13:21
怎么看啊。汇编看不懂诶

51单片机是8位的系统,当一个变量用32位定义时,编译器会对该变量做一系列寄存器操作。
你期望延时100us,但实际上会长很多。
yao1
4楼-- · 2020-02-07 07:44

//**********************************************************************************
//延时函数, 对于12MHz时钟, 例z=10,则大概延时10ms.  

void delay(uint z)
{
    uint x,y;
    for(x=z;x>0;x--);
        for(y=124;y>0;y--);  
}     

//**********************************************************************************

//**********************************************************************************
//初始化DS18B20
//让DS18B20一段相对长时间低电平, 然后一段相对非常短时间高电平, 即可启动
void dsreset(void)        
{  //对于11.0592MHz时钟, unsigned int型的i, 作一个i++操作的时间大于?us
    uint i;
    ds=0;
    i=103;    //拉低约800us, 符合协议要求的480us以上
    while(i>0)i--;
    ds=1;    //产生一个上升沿, 进入等待应答状态
    i=4;
    while(i>0)i--;
}
//**********************************************************************************

//**********************************************************************************
//向DS18B20读取一位数据
//读一位, 让DS18B20一小周期低电平, 然后两小周期高电平,
//之后DS18B20则会输出持续一段时间的一位数据
bit tempreadbit(void)
{  
    uint i;
    bit dat;
    ds=0;
    i++;      //i++起延迟作用  延时约8us, 符合协议要求至少保持1us
    ds=1;
    i++;i++;  //延时约16us, 符合协议要求的至少延时15us以上
    dat=ds;
    i=8;
    while(i>0)i--; //延时约64us, 符合读时隙不低于60us要求
    return(dat);
}
//**********************************************************************************

//**********************************************************************************
//读取一字节数据, 通过调用tempreadbit()来实现
uchar tempread(void)      
{
    uchar i,j,dat;
    dat=0;
    for(i=1;i<=8;i++)
    {
        j=tempreadbit();
        dat=(j<<7)|(dat>>1);  //读出的数据最低位在最前面,这样刚好一个字节在dat里   
    }
    return(dat);
}
//**********************************************************************************

//**********************************************************************************
//向DS18B20写入一字节数据
void tempwritebyte(uchar dat)   
{
    uint  i;
    uchar j;
    bit testb;
    for(j=1;j<=8;j++)
    {
        testb=dat&0x01;
        dat=dat>>1;
        if(testb)    //写"1", 将ds拉低15us后, 在15us~60us内将ds拉高, 即完成写1
        {
            ds=0;
            i++;i++;  //拉低约16us, 符号要求15~60us内
            ds=1;
            i=8;while(i>0)i--;  //延时约64us, 符合写时隙不低于60us要求
        }
        else      //写"0", 将ds拉低60us~120us
        {
            ds=0;  
            i=8;while(i>0)i--;  //拉低约64us, 符号要求      
            ds=1;
            i++;i++; //整个写0时隙过程已经超过60us, 这里就不用像写1那样, 再延时64us了   
        }   
    }
}
//**********************************************************************************

//**********************************************************************************
//向DS18B20发送温度转换命令
void tempchange(void)   
{
    dsreset();   //初始化DS18B20, 无论什么命令, 首先都要发起初始化
    delay(1);    //延时1ms, 因为DS18B20会拉低ds 60~240us作为应答信号
    tempwritebyte(0xcc);   //写跳过读ROM指令
    tempwritebyte(0x44);   //写温度转换指令
}
//**********************************************************************************

//**********************************************************************************
//向DS18B20发送读取数据命令  获取当前温度值
uint tmp()  
{
    float tt;
  //  int tt;
    uchar low,high;
    dsreset();
    delay(1);
    tempwritebyte(0xcc); //写入跳过序列号命令字 Skip Rom
    tempwritebyte(0xbe); //写入读取数据令字 Read Scratchpad
    //连续读取两个字节数据
    low =tempread();   //读低8位
    high=tempread();   //读高8位
    temp=high;   //高8位赋予temp
    temp<<=8;    //将高低两个字节合成一个整形变量 temp为16位 高8位左移8位 低8位补0
    temp=temp|low;  //高低8位相或=temp为16位=高8位+低8位

    xs=low&0x0f; //低4位为小数部分
    tt=temp>>4; //整数部分
  //  temp=tt*100+xs*100/16;  

     temp=tt*10+((((xs*5)>>2)+1)>>1);
  //   **************************************
   //y=(((x*5)>>2)+1)>>1;
//等效y=((x*5)/4+1)/2;
//再等效y=((x*20)/16+1)/2;
//y=2*(x*10/16+0.5)/2;
//实际上y=x*10/16+0.5;  //+0.5为 四舍五入
//  **********************************************
  // tt=temp*0.0625;  //DS18B20的默认分辨率12位, 精确度为0.0625度, 即读回数据的最低位代表0.0625度   
// temp = tt * 100 + (temp > 0 ? 0.5 : -0.5); //大于0加0.5, 小于0减0.5  
//   temp=tt*100;  //将它放大100倍, 使显示时可显示小数点后两位(t=11.0625,计算得temp = 1106, 即11.06 度)
    return temp; //temp是整型
}
//**********************************************************************************

//**********************************************************************************
//显示函数
void display(uint t)
{
     unsigned char a1=0,a2=0,a3=0,a4=0;        //临时变量
     static int k=0;   
      
      a1=t/1000;     //取t的千位//
      a2=t%1000/100;//取t的百位//
      a3=t%100/10; //取t的十位//
      a4=t%10;    //取t的个位//

      if     (k==0){A1=1;A2=0;A3=0;A4=0;P0=yima[a1];}
      else if(k==1){A1=0;A2=1;A3=0;A4=0;P0=yima[a2];}
      else if(k==2){A1=0;A2=0;A3=1;A4=0;P0=yima[a3];}
      else if(k==3){A1=0;A2=0;A3=0;A4=1;P0=yima[a4];}   

      k++;
      if(k>3) k=0;
}      
yao1
5楼-- · 2020-02-07 11:17
 精彩回答 2  元偷偷看……
BXAK
6楼-- · 2020-02-07 14:31
yesno 发表于 2012-5-19 16:09
测试的怎么样啦

官网提供的 STCMCU DS18B20.rar (39.59 KB, 下载次数: 10) 2012-5-19 17:29 上传 点击文件名下载附件

一周热门 更多>