DS18B20多点温度检测C51程序

发布者:ShiningSmile最新更新时间:2016-11-18 来源: eefocus关键字:DS18B20  多点温度检测  C51程序 手机看文章 扫描二维码
随时随地手机看文章

#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<(j+1)*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]);

   }

  }

 }

}


关键字:DS18B20  多点温度检测  C51程序 引用地址:DS18B20多点温度检测C51程序

上一篇:ds1302-at89c51头文件+c51驱动
下一篇:128*64液晶屏(ST7920)C51驱动

推荐阅读最新更新时间:2024-03-16 15:21

DS18B20温度格式转换
#include stdio.h #include stdint.h /* DS18B20温度格式转换 */ uint8_t tempIntPart ; uint8_t tempDecPart ; float convertToFloat(int16_t temperature) { float temp; temp = (float)temperature; temp /= 16.0f; return temp; } void splitIntPart(int16_t temperature) { uint8_t i; if (temperature & 0x8000) temper
[单片机]
C51单片机LCD1602驱动程序
LCD1602简介 LCD1602字符型液晶(每行显示16个字符,一共可以显示两行)——一种专门用来显示字母、数字、符号的点阵型液晶模块。它由若干个5x7或者5x10的点阵字符组成,每个点阵字符位都可以用来显示一个字符,每位之间有一个点距的间隔,每行之间也有间隔,起到了字符间距和行间距的作用,正因为如此,所以它不能很好的显示图片。 1602的引脚 我们只需要关注以下几个管脚: 3 脚:VL,液晶显示偏压信号,用于调整LCD1602 的显示对比度,一般会外接电位器用以调整偏压信号,注意此脚电压为0时可以得到最强的对比度。 4 脚:RS,数据/命令选择端,当此脚为高电平时,可以对1602 进行数据字节的传输操作,而为电平时,
[单片机]
<font color='red'>C51</font>单片机LCD1602驱动<font color='red'>程序</font>
一套热计量仪表温度控制系统的设计方案
0引言 随着生活水平的提高,人们对生活的环境提出了更高的要求,良好的温度控制对提高人民生活质量起着非常重要的作用。 特别是北方,城镇居民冬季一般采取集中供暖,采暖质量的好坏,直接关系到人们生产生活。目前我国绝大数的地区采暖缺少有力的监控系统,无法完成按需采暖以及实现对温度数据的有效控制,从而造成了对供暖热能的浪费。随着人们节约能源意识的逐步增强,迫切需要一种操作简单、节能环保和高效利用的智能热计量仪表温度控制系统。 1系统总体设计 根据热量与流量、温度之间的关系: Q=CM(T1-T0)式中,Q为散热器的散热量(单位:J);C为水的比热4.2*103J/kg°C ;T1-T0为散热器进出口的温差(°C);M为流经散热器的水的流量(L
[单片机]
一套热计量仪表温度控制系统的设计方案
51单片机驱动DS18B20温度传感器程序及心得
关于DS18B20温度传感器,在没有硬件设备的辅助下,写内部程序有些困难,因为看不到实际信号波形。对于单片机,我。。。渐渐的有些心灰意冷。。虽然掌握了1_WIRE总线,却少了很多喜悦,下雨了。。。它是我的爱好,我付出了很多,可是我看不到实际的前景。。以我个人之力,要步入尖端芯片领域,很困难,在这里,采棉花是个普遍性的大问题,大型机械设备缺陷很多,如果以微控制芯片提高精度,我想效益会相当可观,可是技术瓶颈难以逾越。。。硬件研发,失败了,所有投入赴之东流,成功了,回报丰厚。现在,各行各业都处于饱和,没有成熟先进的技术,很难有立足之地,,,,,我开始重新审视我的选择。。。。。。艰难。。 /* 建立时间: 2013年5月2日;
[单片机]
PID调节C51程序(1)
关于PID 比例调节作用:是按比例反应系统的偏差,系统一旦出现了偏差,比例调节立即产生调节作用用以减少偏差。比例作用大,可以加快调节,减少误差,但是过大的比例,使系统的稳定性下降,甚至造成系统的不稳定。 积分调节作用:是使系统消除稳态误差,提高无差度。因为有误差,积分调节就进行,直至无差,积分调节停止,积分调节输出一常值。积分作用的强弱取决与积分时间常数Ti,Ti越小,积分作用就越强。反之Ti大则积分作用弱,加入积分调节可使系统稳定性下降,动态响应变慢。积分作用常与另两种调节规律结合,组成PI调节器或PID调节器。 微分调节作用,微分作用反映系统偏差信号的变化率,具有预见性,能预见偏差变化的趋势,因此能产生超前的控制作用,在偏差还
[单片机]
基于AT89S52和DS18B20的温度显示报警系统
  引言   温度是一种最基本的环境参数,人们的生活环境与温度息息相关,温度测量也被人们所异常关注。因此,研究温度的测量方法和装置具有重要意义,温度测控技术也在各个领域应用越来越广泛。采用单片机对温度进行控制,不仅具有控制方便和组态简单的优点,而且可以提高被控温度的技术指标。本文介绍了一款由单片机AT89S52 和新型的智能集成温度 传感器 DS18B20 以及LCD 显示器等部件实现的温度测量及报警系统。同时在设计方面做了功能的扩展,键盘是用来调时和温度查询,功能较强,可以设置上下限报警温度,且测量准确、误差小。单片机可把由DS18B20、DS1302 读来的数据利用软件来进行处理,从而把数据传输到显示模块,实现温度、日历的显
[单片机]
基于AT89S52和<font color='red'>DS18B20</font>的温度显示报警系统
DS18B20和PIC单片机通讯源程序
ORG PIC54 GOTO MAIN ORG 0 ;---------------------- ;---------------------------- DELAY22 MOVLW D'200' ; DELAY 2*250=500mS MOVWF COUNT1 DE32 MOVLW D'250' ; 8*250=2mS MOVWF COUNT2 DE42 NOP ; 1+2+1=5uS DECFSZ COUNT2,1 GOTO DE42 DECFSZ COUNT1,1 GOTO
[单片机]
实时时钟芯片DS1302的C51程序列子
/*********************************************************************/ /* 实时时钟模块 时钟芯片型号:DS1302 */ /*/ /*********************************************************************/ sbit T_CLK = P2^7; /*实时时钟时钟线引脚 */ sbit T_IO = P1^4; /*实时时钟数据线引脚 */ sbit T_RST = P1^5; /*实时时钟复位线引脚 */ /*****************************************
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved