#include <reg51.h>
#include <intrins.h>
#define Disdata P0 //段码输出口
#define discan P3 //扫描口
#define uchar unsigned char
#define uint unsigned int
unsigned char data dis_digit;
unsigned char key_s, key_v,i;
uchar dis[4]={0x00,0x00,0x00,0x00};
uchar dis1[4]={0x00,0x00,0x00,0x00};
unsigned char code dis_code[11]={0xc0,0xf9,0xa4,0xb0, // 0, 1, 2, 3
0x99,0x92,0x82,0xf8,0x80,0x90, 0xff};// 4, 5, 6, 7, 8, 9, off
unsigned char dis_buf[8]; // 显示缓冲区
char sec_bcd[8]; // 秒计数值, BCD码
unsigned char dis_index,num=0,num1=0,num2=0,a=0,aa=0; //
unsigned char key_times; // K1 按下次数
uint h;
sbit DQ=P3^7; //温度输入口
sbit DIN=P0^7; //LED小数点控制
sbit key1=P2^0;
sbit key2=P2^1;
sbit key3=P2^3;
sbit key4=P2^4;
sbit key5=P2^5;
sbit p1=P1^0;
uchar ss=0;
void clr_time();
void update_disbuf();
bit scan_key();
void proc_key();
void delayms(unsigned char ms);
sbit K1 = P2^2;
sbit d1=P2^6;
sbit d2=P2^7;
//*******温度小数部分用查表法**********//
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]={0x01,0x02,0x04,0x08}; // 列扫描控制字
uchar data temp_data[2]={0x00,0x00}; // 读出温度暂放
uchar data display[5]={0x00,0x00,0x00,0x00,0x00};//显示单元数据,共4个数据,一个运算暂存用
//
//
//
/***********11微秒延时函数**********/
//
void delay1(uint t)
{
for(;t>0;t--);
}
//
/***********显示扫描函数**********/
scan()
{
char k=0;
for(k=0;k<4;k++) //四位LED扫描控制
{
Disdata=dis_7[display[k]];
if(k==1){DIN=0;}
discan=scan_con[k];delay1(90);discan=0xff;
}
}
//
//
/***********18B20复位函数**********/
ow_reset(void)
{
char presence=1;
while(presence)
{
while(presence)
{
DQ=1;_nop_();_nop_();
DQ=0; //
delay1(50); // 550us
DQ=1; //
delay1(6); // 66us
presence=DQ; // presence=0继续下一步
}
delay1(45); //延时500us
presence = ~DQ;
}
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; //最低位移出
delay1(6); //66us
val=val/2; //右移一位
}
DQ = 1;
delay1(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;
delay1(6); //66us
}
DQ=1;
return(value);
}
//
/***********读出温度函数**********/
//
read_temp()
{
ow_reset(); //总线复位
write_byte(0xcc); // 发Skip ROM命令
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); // 发转换命令
}
//
/***********温度数据处理函数**********/
work_temp()
{
uchar n=0; //
if(temp_data[1]>127)
{temp_data[1]=(256-temp_data[1]);temp_data[0]=(256-temp_data[0]);n=1;}//负温度求补码
display[4]=temp_data[0]&0x0f;display[0]=ditab[display[4]];
display[4]=((temp_data[0]&0xf0)>>4)|((temp_data[1]&0x0f)<<4);//
display[3]=display[4]/100;
display[1]=display[4]%100;
display[2]=display[1]/10;
display[1]=display[1]%10;
if(!display[3]){display[3]=0x0a;if(!display[2]){display[2]=0x0a;}}//最高位为0时都不显示
if(n){display[3]=0x0b;}//负温度时最高位显示"-"
}
//
/**************主函数****************/
void temp()
{
Disdata=0xff; //初始化端口
discan=0xff;
for(h=0;h<4;h++){display[h]=8;}//开机显示8888
ow_reset(); // 开机先转换一次
write_byte(0xcc); // Skip ROM
write_byte(0x44); // 发转换命令
for(h=0;h<500;h++)
{scan();} //开机显示"8888"2秒
while(1)
{
read_temp(); //读出18B20温度数据
work_temp(); //处理温度数据
for(h=0;h<500;h++)
{scan();} //显示温度值2秒
}
}
//
//*********************结束**************************//
void delay(uint x)
{
uint i,j;
for(i=x;i>0;i--)
for(j=110;j>0;j--);
}
void key()
{
if(key1==0)
{
delayms(10);
if(key1==0)
{ ss=1;
num++;
if(num==99)
num=0;
while(key1==0);
clr_time();
}
}
else if(key2==0)
{
delayms(10);
if(key2==0)
{ ss=1;
if(num==0)
num=99;
num--;
while(key2==0);
clr_time();
}
}
else if(key5==0)
{
delayms(10);
if(key5==0)
{
TR0=0;
temp();
}
}
else if(key3==0)
{
delayms(10);
if(key3==0)
{
{
dis[0]=sec_bcd[0];
dis[1]=sec_bcd[1];
dis[2]=sec_bcd[2];
dis[3]=sec_bcd[3];
}
}
}
else if(key4==0)
{
delayms(10);
if(key4==0)
{
dis_buf[0] = dis_code[dis[0]];
dis_buf[1] = dis_code[dis[1]];
dis_buf[2] = dis_code[dis[2]] & 0x7f; // 加上小数点
dis_buf[3] = dis_code[dis[3]];
}
P3 = 0xff; // 先关闭所有数码管
P0 = dis_buf[dis_index]; // 显示代码传送到P0口
P3 = dis_digit; //
dis_digit = _crol_(dis_digit,1); // 位选通值右移(P20<-P27), 下次中断时选通下一位数码管
dis_index++; //
dis_index &= 0x03; // 8个数码管全部扫描完一遍之后,再回到第一个开始下一次扫描
}
}
void main(void)
{
d1=1;
d2=1;
P0 = 0xff;
P3 = 0xff;
TMOD = 0x11; // 定时器0, 1工作模式1, 16位定时方式
TH1 = 0xdc;
TL1 = 0;
TH0 = 0xFC;
TL0 = 0x17;
clr_time(); //
dis_digit = 0x01; // 初始显示P20口数码管
dis_index = 0; //
key_times = 0;
key_v = 0x01;
IE = 0x8a; // 使能timer0, timer1中断
TR0 = 1;
TR1 = 0;
while(1)
{
if(scan_key())
{
delayms(10);
if(scan_key())
{
key_v = key_s;
proc_key();
}
}
key();
}
}
void clr_time()
{
sec_bcd[0] = 0x0;
sec_bcd[1] = 0x0;
sec_bcd[2] = num%10;
sec_bcd[3] = num/10;
sec_bcd[4] = 0x0;
sec_bcd[5] = 0x0;
sec_bcd[6] = 0x0;
sec_bcd[7] = 0x0;
update_disbuf();
}
bit scan_key()
{
key_s = 0x00;
key_s |= K1;
return(key_s ^ key_v);
}
void proc_key()
{
if((key_v & 0x01) == 0)
{
key_times++;
if(key_times/2 !=(key_times+1)/2 )
{
TR1 = 1;
d1=0;
d2=1;
}
else if(key_times/2 ==(key_times+1)/2)
{
TR1 = 0;
d2=0;
d1=1;
}
}
}
void timer0() interrupt 1
// 定时器0中断服务程序, 用于数码管的动态扫描
// dis_index --- 显示索引, 用于标识当前显示的数码管和缓冲区的偏移量
// dis_digit --- 位选通值, 传送到P2口用于选通当前数码管的数值, 如等于0xfe时,
// 选通P2.0口数码管
// dis_buf --- 显于缓冲区基地址
{
TH0 = 0xFC;
TL0 = 0x17;
P3 = 0xff; // 先关闭所有数码管
P0 = dis_buf[dis_index]; // 显示代码传送到P0口
P3 = dis_digit; //
dis_digit = _crol_(dis_digit,1); // 位选通值右移(P20<-P27), 下次中断时选通下一位数码管
dis_index++; //
dis_index &= 0x03; // 8个数码管全部扫描完一遍之后,再回到第一个开始下一次扫描
}
void js()
{unsigned char i;
for(i = 0; i < 8; i++)
{
sec_bcd[i]--; // 低位加1
if(sec_bcd[i] >-1) // 如果低位满10则向高位进1
break; // 低位未满10
sec_bcd[i] = 9; // 低位满10清0
}
if(sec_bcd[0]==0 && sec_bcd[1]==0 && sec_bcd[2]==0 && sec_bcd[3]==0)
TR1=0;
}
void jsj()
{unsigned char i;
for(i = 0; i < 8; i++)
{
sec_bcd[i]++; // 低位加1
if(sec_bcd[i] < 10) // 如果低位满10则向高位进1
break; // 低位未满10
sec_bcd[i] = 0; // 低位满10清0
}
if(sec_bcd[0]==0x82 && sec_bcd[1]==0 && sec_bcd[2]==0 && sec_bcd[3]==0)
TR1=0;
}
void timer1() interrupt 3
//
{
TH1 |= 0xdc;
if(ss==1){js();}
else i
评论0