📄 wd.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 + -