×
单片机 > 单片机程序设计 > 详情

DS18B20多点温度检测C51程序

发布时间:2025-11-28 发布时间:
|

#include

#include           //_nop_();延时函数用

#define uchar  unsigned char

#define uint  unsigned int

//********************** DS18B20 指令********************* *********   

#define ds18B20_READ_ROM          0x33      // 读ROM指令   

#define ds18B20_MATCH_ROM         0x55      // 匹配ROM指令   

#define ds18B20_SKIP_ROM          0xCC      // 跳过ROM指令   

#define ds18B20_SEARCH_ROM        0xF0      // 搜索ROM指令   

#define ds18B20_ALARM_SEARCH      0xEC      // 报警搜索指令   

#define ds18B20_WRITE_SCRATCHPAD  0x4E      // 写暂存寄存器指令   

#define ds18B20_READ_SCRATCHPAD   0xBE      // 读暂存寄存器指令   

#define ds18B20_COPY_SCRATCHPAD   0x48      // 复制暂存寄存器指令   

#define ds18B20_CONVERT_T         0x44      // 启动温度转换指令   

#define ds18B20_RECALL_E2         0xB8      // 重新调出E2PROM的数据 

 

sbit  DQ=P0^7;     //将P0.7口模拟时钟数据口   

bit  flag=0;

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

 温度小数部分用查表法

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

uchar code ditab[16]=

{0x00,0x01,0x01,0x02,0x03,0x03,0x04,0x04,0x05,0x06,0x06,0x07,0x08,0x08,0x09,0x09};

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

 延时函数

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

void delay_us(unsigned int us)   

{

    unsigned int i;

  us=us/7;         //1/7是STC89C58在22.1184MHz晶振

  for( i=0;i

}

void delay_ms(unsigned int ms)   

{

    unsigned int i,j;

  for( i=0;i

  for(j=0;j<332;j++);  //332是STC89C58在22.1184MHz晶振

}

/***********读出温度函数************

输入:无

输出:zs+xs,格式为123.4

      即为浮点型数的温度格式

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

/***********18B20复位函数**********/

void Init18b20 (void)

{

 DQ=1;;_nop_();_nop_();

 DQ=0;delay_us(600);  //要大于480us,小于960us

 DQ=1;delay_us(60);  //要大于15us--60us 

 

 if(DQ==0)

    flag = 1;      //detect 1820 success!

 else

    flag = 0;       //detect 1820 fail!

 delay_us(200);   //要大于60us,小于240us 

 DQ = 1;

}

/**********18B20写命令函数*********

 向 1-WIRE 总线上写一个字节

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

void write_byte(uchar val)

{

 uchar i;

 for (i=8; i>0; i--)

 {

  DQ=1;_nop_();_nop_();

  DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us

  DQ = val&0x01;      //最低位移出

  delay_us(60);       //66us

  val=val>>1;         //右移一位

 }

 DQ = 1;

 delay_us(10);

}

/*********18B20读1个字节函数********

 从总线上读取一个字节

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

uchar read_byte(void)

{

 uchar i;

 uchar value = 0;

 for (i=8;i>0;i--)

 {

  DQ=1;_nop_();_nop_();

  value>>=1;

  DQ = 0;_nop_();_nop_();_nop_();_nop_();   //4us

  DQ = 1;_nop_();_nop_();_nop_();_nop_();   //4us

  if(DQ)value|=0x80;

  delay_us(60);      //60us

 }

 DQ=1;

 delay_us(10);

 return(value);

}

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

 串口中断服务子程序

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

static void com_int(void) interrupt 4

{

 if (RI)

 {

  RI = 0;

  if (SBUF == 0x0d)

  {

   ISP_CONTR = 0x60; // 软复位到ISP区

  }

 }

}

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

   串口发送一个字符

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

void com_send_dat( unsigned char dat)

{

    SBUF=dat;

 while (TI== 0);

 TI= 0 ;

}

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

   串口初始化

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

void init_com( void )

{

 SCON=0x50 ;   //SCON: serail mode 1, 8-bit UART, enable ucvr   //UART为模式1,8位数据,允许接收

    TMOD|=0x20 ;  //TMOD: timer 1, mode 2, 8-bit reload            //定时器1为模式2,8位自动重装

 TH1=0xfa ;     //Baud:19200 fosc="22.1184MHz

 TL1=0xfa;

 PCON|=0x80;  //SMOD=1;波特率加倍;  

 ES=1;      //Enable Serial Interrupt

   TR1 = 1 ;       // timer 1 run

}

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

   搜索DS18B2064位ROM

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

uchar search_rom(uchar *p)

{

 uchar dat,i,j,k,n,value,number=1,number_temp=1,clash_number=0;

 bit clash_flag=0;

 for(n=0;n

 {

  flag=0;

  while(!flag)Init18b20();  //复位ds18b20

  write_byte(ds18B20_SEARCH_ROM);     //搜索ROM命令

  for(i=0;i<8;i++)    //8个字节第ROM

  {

   dat=0;

   for(j=0;j<8;j++)   //8位数据

   {

    value=0;

    for(k=0;k<2;k++)  //一位数据位,一位补码位

    {

     DQ=1;_nop_();_nop_();

     value<<=1;

     DQ = 0;_nop_();_nop_();_nop_();_nop_();   //4us

     DQ = 1;_nop_();_nop_();_nop_();_nop_();   //4us

     if(DQ)value|=0x01;

     delay_us(60);      //60us

    }

    switch(value)   //根据回复第两位数据来发出相应操作

    {

     case 0x00:    //总线冲突

     if(n==0)  

     {

      number++;

      number_temp=number;

      dat=dat>>1;

      DQ = 1;_nop_();_nop_();

      DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us

      DQ = 0;         //最低位移出

      delay_us(60);     //60us

     }

     else      //第二次以上循环时

     {

      clash_number++;

      if(clash_number>=(number_temp-1))

      {

       if(!clash_flag)

       {

        clash_flag=1;

        dat=dat>>1;

        dat=dat|0x80; 

        DQ = 1;_nop_();_nop_();

        DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us

        DQ = 1;       //最低位移出

        delay_us(60);   //60us          

       }

       else 

       {       

        clash_flag=0;

        number_temp--;

        dat=dat>>1;

        DQ = 1;_nop_();_nop_();

        DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us

        DQ = 0;       //最低位移出

        delay_us(60);   //60us          

       }

      }     

     }

     break;

     case 0x01:

     dat=dat>>1;

     DQ = 1;_nop_();_nop_();

     DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us

     DQ = 0;          //最低位移出

     delay_us(60);         //66us

     break;

     case 0x02:

     dat=dat>>1;

     dat=dat|0x80; 

     DQ = 1;_nop_();_nop_();

     DQ = 0;_nop_();_nop_();_nop_();_nop_();_nop_();//5us

     DQ = 1;          //最低位移出

     delay_us(60);         //66us

     break;

     case 0x03:i=8;j=8;break;

     default:break;

    }    

   } 

   *p++=dat;

  }

 }

 return number;

}

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

   多个DS18B20测试主函数

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

void main()

{

 uchar i,j,dat,amount;

 uchar ROM_data[16];       //读出ROM暂存

 uchar temp_data[2];       //读出温度暂存

 uchar temperature[7]={0,0,0,'.',0,'?,'?}; //要发送的数据缓存

 init_com();

 EA=0;

 while(1)

 {

  amount=search_rom(ROM_data);

  flag=0;

  while(!flag)Init18b20();

  write_byte(ds18B20_SKIP_ROM);    // 跳过ROM命令

  write_byte(ds18B20_CONVERT_T);    // 发转换命令

  EA=1;

  delay_ms(1000);

  EA=0;

   for(j=0;j

  {

   flag=0;

   while(!flag)Init18b20();

   write_byte(ds18B20_MATCH_ROM);   // 符合ROM命令

   for(i=j*8;i

   {

    write_byte(ROM_data[i]);  

     }

   write_byte(ds18B20_READ_SCRATCHPAD);//发读命令

   temp_data[0]=read_byte();    //温度低8位

   temp_data[1]=read_byte();     //温度高8位

 

   temperature[4]=ditab[temp_data[0]&0x0f]+0x30;   //小数位

   dat=((temp_data[0]&0xf0)>>4)|((temp_data[1]&0x0f)<<4); //整数位

   for(i=0;i<3;i++)

   {

    temperature[2-i]=dat%10+0x30;

    dat=dat/10;

   }

   

   for(i=0;i<7;i++)     //串口发送温度数据

   {

    com_send_dat(temperature[i]);

   }

  }

 }

}





『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
cortex M3S811学习笔记-GPIO-INT