⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 wd.c

📁 采集温度源码 准确能用!! 可以下栽用用 -) 51单片机就可以实现,单线式采集
💻 C
字号:
#include <aduc812.h>
#include <intrins.h>
sbit led0=P3^4;             //P3.4~P3.7用作4位LED的位选线
sbit led1=P3^5;
sbit led2=P3^6;
sbit led3=P3^7;
sbit DQ = P2^4;                         //P2.4用作DS18B20的数据线DQ 
float data TMP[2]={0,0};           //读取后的2个温度值,将其除以2即可得出实际温度;
unsigned char data f[2]={0,0};    //结果是否为负温,"0"为正温,"1"为负温。
unsigned char data disp_buf[4]={0,0,0,0}; //4位数码管对应的值放入该缓冲区
unsigned char data dot_position=0;
unsigned char data chno=0;          //对应某路DS18B20
//存各路DS18B20的地址序列号,为便于调试,只设计了2路,可以扩充到8路或更多
unsigned char code SN[2][8]={ {16, 62,148,60,0,0,0, 247},{16,229,146,60,0,0,0, 87} };
//数字0~9和通道提示符A~H的段码
unsigned char code seg_table[ ]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,
0x77,0x7c,0x39,0x5e,0x79,0x71,0x6f,0x76};     
unsigned char code CH[ ]={10,11,12,13,14,15,16,17};      //通道提示符的段码偏移量

//将0.00~999之间的浮点数转为单个数码,并送显示缓冲区和返回小数点的位置
void ftochar(float valp)
{
if(valp<10.0) 
{
dot_position=1;
valp *=100.0;
}
else if((valp>=10.0)&&(valp<100.0))
{
dot_position=2;
valp *=10.0;
}
else if((valp>=100.0)&&(valp<1000.0)) dot_position=3;
disp_buf[1]=(int)valp/100;
disp_buf[2]=((int)valp%100)/10;
disp_buf[3]=((int)valp%100)%10;
}

//延时15微妙的函数
void delay(unsigned char n)
{
do    {
_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();    //_nop_()的头文件为intrins.h
_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();
n--;
}while(n);
}
//DS18B20复位函数,按复位时序进行设计
void ow_reset(void)
{
DQ = 0;                // DQ置为低电平
delay(36);             // 保持480μs
DQ = 1;                // DQ置为高电平
delay(24);             // 延时,等DS18B20输出低电平
} 
//DS18B20读位函数,按读位时序进行设计
unsigned char read_bit(void)
{
unsigned char i;
DQ = 0;                       // DQ置为低电平
DQ = 1;                       // DQ置为高电平
for (i=0; i<5; i++);        // 延时 15μs 
return(DQ);                 // 返回DQ 线的电平状态
}
// DS18B20写位函数,按写位时序进行设计
void write_bit(char bitval)
{
DQ = 0;                       // DQ置为低电平
if(bitval==1) DQ =1;     // 如果写1则DQ置为高电平
delay(6);                      // 延时以维持电平状态
DQ = 1;                       // DQ置为高电平
}     

// 从DS18B20读取字节的函数
unsigned char read_byte(void)
{
unsigned char i;
unsigned char value = 0;
for (i=0;i<8;i++)
{
if(read_bit()) value|=0x01<<i;      //调用读位函数,读出的8个位移位成1个字节
delay(11);                                  //延时以读余下的位
}
return(value);
}
//写字节到DS18B20的函数
void write_byte(char val)
{
unsigned char i;
unsigned char temp;
for (i=0; i<8; i++)                //每次写1位,1个字节分8次完成
{
temp = val>>i;       
temp &= 0x01; 
write_bit(temp);                  //调用写位函数
}
delay(10);                            //延时
}

// 从DS18B20读物温度代码
void  read_temp () 
{
unsigned char i,j; 
unsigned char a,b; 
int mr;
for(j=0;j<2;j++)            //为便于调试,仅以2路为例,改循环次数即可扩充到8路或更多,
{ 
ow_reset();           //调用复位函数
delay(20);
write_byte (0x55); //发送ROM匹配命令 
for(i=0;i<8;i++) 
{ 
write_byte(SN[j][i]); //发送64位序列号 
}
write_byte (0xbe);        //发送读取暂存寄存器的命令 
a = read_byte();     //连续读取两位温度,余下数据没有读,实际使用时应读出所有数
b = read_byte();     //据,并进行校验,以提高可靠性
mr=b*256+a;
if((mr&0xf800)!=0) mr=-mr+1;
TMP[j]=mr*0.5;
} 
}

//定时器T0中断函数,每中断1次,显示1位数码管
void Time_disp(void) interrupt 1
{
static unsigned char dispno=0;      //数码管位号
TH0=0xee;                                 //主频为11.0592,定时5ms的时间常数为EE00H
TL0=0x00;
P3|=0xf0;
P0=seg_table[disp_buf[dispno]];          //查当前数码管的显示数字对应的段码
if(dispno==dot_position) P0|=0x80;      //当前位有小数点,则段码最高位置1
switch(dispno)                            //根据当前显示的数码管,接通位选线
{
case 0 :   led0=0;    break;
case 1 :   led1=0;    break;
case 2 :   led2=0;    break;
case 3 :   led3=0; break;
}
dispno++;
if(dispno==4) dispno=0;
}

//定时器T1中断服务函数,每50ms中断1次
void Timer1(void) interrupt 3
{
static unsigned int count;
TH1=0x4c;            //50ms对应的时间常数为4C00H
TL1=0x00;
count++;
if(count>=20)        //中断20次即为1秒
{
count=0;
ftochar(TMP[chno]);     //当前通道对应的温度值转换为单个数码送显示缓冲区
disp_buf[0]=CH[chno];  //当前通道的提示符的段码偏移量送显示缓冲区首地址
chno++;
if(chno= =2) chno=0;    //修改此判断对应的数值,即可扩充到8路或更多
}
}
//主函数
main( ) 
{ 
TMOD=0x11;        //定时器T0和T1按方式1工作
EA=1;
ET0=1;
ET1=1;
TH0=0xee;            //5ms对应的时间常数
TL0=0x00;
TH1=0x4c;            //50ms对应的时间常数
TL1=0x00;
TR0=1;
TR1=1;
do    { 
ow_reset( );                  //复位DS18B20
write_byte(0xcc);
write_byte(0x44);   //启动1820
read_temp( );               //调用读取温度的函数,结果存于TMP[ ]数组中
}while(1); 
} 

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -