用单片机设计一个温度采集系统

2019-07-16 07:06发布

要求如下:传感器用DS18B20型号的,单片机用AT89C51型号的,使用数码管显示(显示有一位小数点)。那位好心人有这个程序和原理图。分享一下,非常感谢啊!
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
8条回答
暮鼓
2019-07-17 05:48
#include "reg51.h"
#include "intrins.h"                            //_nop_();延时函数用
#define  Disdata P0                             //段码输出口
#define  discan  P2                             //扫描口
#define  uchar unsigned char
#define  uint  unsigned int
sbit     DQ=P3^3;                               //温度输入口
sbit     DIN=P0^7;                              //LED小数点控制
uint     h;
uchar flag;
//**************温度小数部分用查表法***********//
uchar code ditab[16]=
{0x00,0x01,0x01,0x02,0x03,0x03,0x04,0x04,0x05,0x06,0x06,0x07,0x08,0x08,0x09,0x09};
//
uchar code dis_7[12]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xff,0xbf};
//共阳LED段码表        "0"  "1"  "2"  "3"  "4"  "5"  "6"  "7"  "8"  "9" "不亮" "-"              
uchar code scan_con[4]={0x7f,0xbf,0xdf,0xef};    //列扫描控制字
uchar data temp_data[2]={0x00,0x00};             //读出温度暂放
uchar data display[5]={0x00,0x00,0x00,0x00,0x00};     //显示单元数据,共4个数据和一个运算暂用

//
//
//
/***********11微秒延时函数**********/
//
void delay(uint t)
{
for(;t>0;t--);
}
//
/***********显示扫描函数**********/
scan()
{
char k;
    for(k=0;k<4;k++)        
     {
         Disdata=0xff;
      Disdata=dis_7[display[k]];
      if(k==1){DIN=0;}
      discan=scan_con[k];delay(90);
          discan=0xff;
     }
}
//
//
/***********18B20复位函数**********/
ow_reset(void)
{
char presence=1;
while(presence)
{
while(presence)
{
DQ=1;_nop_();_nop_();
DQ=0;  
delay(50);
DQ=1;   
delay(6);
presence=DQ;
   }
delay(45);   
presence = ~DQ;
}
DQ=1;
}
//
//
/**********18B20写命令函数*********/
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(6);         
val=val/2;         
}
DQ = 1;
delay(1);
}
//
/*********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(6);           //66us
}
DQ=1;
return(value);
}
//
/***********读出温度函数**********/
//
read_temp()
{
ow_reset();       //总线复位
write_byte(0xCC); //
write_byte(0xBE); // 发读命令
temp_data[0]=read_byte(); //温度低8位
temp_data[1]=read_byte(); //温度高8位
ow_reset();
write_byte(0xCC); // Skip ROM
write_byte(0x44); // 发转换命令
}
//
/***********温度数据处理函数**********/
void work_temp()
{
uchar n=0;
uchar doth,dotl;
uchar flag3=1,flag2=1;      
if((temp_data[1]&0xf8)!=0x00)
{
   temp_data[1]=~(temp_data[1]);
   temp_data[0]=~(temp_data[0])+1;
   n=1;
   flag=1;
}
if(temp_data[0]>255)
{
temp_data[1]++;
}
display[4]=temp_data[0]&0x0f;
display[0]=ditab[display[4]];
doth=display[0]/10;
dotl=display[0]%10;
display[4]=((temp_data[0]&0xf0)>>4)|((temp_data[1]&0x07)<<4);
display[3]=display[4]/100;
display[2]=display[4]/10%10;
display[1]=display[4]%10;
if(!display[3])
   {
    display[3]=0x0a;
    flag3=0;
    if(!display[2])
    {
     display[2]=0x0a;
     flag2=0;
    }
   }//最高位为0时都不显示
if(n)
{
   display[3]=0x0b;//负温度时最高位显示"-"
   flag3=0;
}
}

main()
{
Disdata=0xff;   
discan=0xff;
for(h=0;h<4;h++){display[h]=8;}
ow_reset();      
write_byte(0xCC);
write_byte(0x44);
for(h=0;h<500;h++)
   {scan();}         
while(1)
{
read_temp();         
work_temp();         
   scan();     }
}

一周热门 更多>