我是电子制造喜好者,最近从网上看到这款单片机驱动的发光二极管做的数码管时钟,以下是作者的介绍“曾经做的数码管时钟,总觉得太小了,远处看不清,就用发光二极管自己制造了一个数码管,这样就亮多了。技能的含量不是很高,可是费功夫,单片机程序是在书上看到的,作了一些小修正。硬件中没有用DS1302,单片机晶振用12M(程序以12M晶振进行核算的),走时比用DS1302还要准。”特此搜集转载。
接口:
数码管:P0^0~P0^7
位选:P2^0~P2^3
设置键P1^5~P1^7(短按:设置显现时刻,长按:设置闹钟时刻)
加1键P1^6
减1键P1^7
BeepP3^7
//作者:whw8099
#include
#include
#define uchar unsigned char
#define uintunsigned int
#defineKSET0x60 //设置当时时刻键
#defineKSET_LONG0x61//设置闹铃时刻键
#defineKINC0x50 //加1键
#defineKDEC0x30//减1键
sbitSPK=P3^7;//蜂鸣器操控引脚
uchar code MAX[2]={24,60};//时、分的最大值
uchar code segtab[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,
0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e,0x89,0x8c,0xff};
uchar dbuf[4]={0,0,0,0};//4位显现缓存
uintcount;//用于累加守时器T0的溢出中止次数,取值为0到4000
uchar TIme[3]={0,0,0}; //当时时、分、秒
uchar ala[2]={0,0};//报警时、分
ucharstate;//state=0:正常走时方法;
//state=1:设置方法;
//state=2:正在闹铃
bitupdate;//时刻更改标志
uchar flash=00;//闪耀操控字
uchar disp_on=0xf0;//显现操控位
void delay(void)
{ uchar i;
for(i=0;i《200;i++);
}
void hexbcd(uchar *tp)
{dbuf[3]=tp[0]/10;//将时拆分为BCD码送显现缓存
dbuf[2]=tp[0]|0x80;
dbuf[1]=tp[1]/10;//将分拆分为BCD码送显现缓存
dbuf[0]=tp[1];
}
void disp(void)
{ staTIc ucharcn=0;
uchar n,bsel;
bsel=0xfe;//开始点亮最低位
for(n=0;n《4;n++)
{ P2=bsel|disp_on;//位选口
P0=segtab[dbuf[n]&0x7f];//显现缓冲单元的数据查出字段码表
if((dbuf[n]&0x80)!=0)
P0=P0&0x7f;
bsel=_crol_(bsel,1);//预备显现下一位
delay();//每位显现约1毫秒
P0=0xff;//平息数码管,避免当时位在下一方位显现出来
}
cn++;//每显现一遍,cn加1
if(cn==200)
{ disp_on=disp_on^flash;
dbuf[2]=dbuf[2]^0x80;
cn=0;
}
}
voidtime0(void) interrupt 1using 1
{ staTIc uintn=0,m=0;
count++;//中止次数加1
if(count==4000)//满1s
{ count=0;//count清0
TIme[2]++;//秒加1
if(time[2]==60)//满60秒
{ time[2]=0;//秒清0
time[1]++;//分加1
if(state!=1) update=1; //更新标志
if(time[1]==60)//满60分
{ time[1]=0;//辨明0
time[0]++;//时加1
if(time[0]==24)//满24小时
time[0]=0;//时清0
}
}
if((state==0)&&(time[2]==0))//当时为走时状况且刚满1分(60秒)
{ if((time[0]==ala[0])&&(time[1]==ala[1])) //闹铃时刻到
{ flash=0x0f;//当时时刻闪耀
disp_on=0xf0;
state=2;//当时状况设置为正在闹铃
dbuf[2]=dbuf[2]|0x80;//将时与分的分隔符显现出来
}
}
}
if(state==0x02)//假如当时正在闹铃
{ if(count%2==0)//假如满500?s(用来发生频率为1KHZ的方波)
{n++;
if(n《500)//在1秒的前1/4秒内
SPK=~SPK; //蜂鸣器发声
else
SPK=1;//发生3/4秒的中止
if(n==2000)//假如满1 S(每次鸣叫距离1S)
{n=0;//n清0,从头再由0计到2000(1S)
m++;//n计满2000(1S)m加1
if(m==6)//假如满6秒
{ state=0;//闹铃时刻完毕
m=0;//m清0
flash=0x00;//康复正常显现
disp_on=0xf0;
}
}
}
}
}
uchar getkey(void)
{ ucharkey;
uchart;
disp();
if((key=P1&0x70)==0x70) return 0xff;
for(t=0;t《5;t++) disp();
if((key=P1&0x70)==0x70) return 0xff;
while((P1&0x70)!=0x70)//检测按键时刻
{ disp();
if(t《250) t++;
}
if((t》200)&&(key==0x60)) return 0x61;
return key;
}
void set_time(uchar *tp)
{ uchar p=0;
uchar k;
state=1;//当时为设置状况
flash=0x0c;//LED显现器最高2位(时)闪耀
while(1)
{ hexbcd(tp);//即将设置的时、分转换成BCD码送显现缓存
k=getkey();//读取按键键值
if(k==KSET)//假如是SET键
{ p++;//调整下标p,使其指向是时或分
p=p&0x01;//避免下标出界
disp_on=0xf0;//康复正常显现
if(p==0x00) //假如指向的是时
flash=0x0c;//LED显现器的高2位闪耀
else
flash=0x03;//LED显现器的低2位闪耀
}
else if(k==KINC)//假如是加1键
{ tp[p]++;//时/分加1
if(tp[p]==MAX[p]) tp[p]=0;//查看时/分是否超出范围
}
else if(k==KDEC)//假如是减1键
{ tp[p]–;//时/分减1
if(tp[p]==0xff) tp[p]=MAX[p]-1;//查看时/分是否超出范围
}
else if(k==KSET_LONG)//假如长按SET键,预备回来
break;
}
hexbcd(time);//康复当时时刻的显现
flash=0x00;//LED显现器康复正常显现
disp_on=0xf0;
state=0;//当时状况康复为正常走时状况
}
voidmain(void)
{
uchark;
state=0;//正常显现时钟方法
count=0;//计数器清0
flash=0x00;//显现器正常显现
disp_on=0xf0;
time[0]=12;//初始化当时时、分、秒
time[1]=25;
time[2]=55;
ala[0]=12;//初始化 时、分
ala[1]=26;
hexbcd(time); //将当时时刻转换为BCD码到显现缓存
TMOD=0x02; //T0作业于守时方法2
TH0=-250;//T0的守时时刻为250?s
TL0=-250;
ET0=1;//答应T0中止
TR0=1;//发动守时
EA=1;//开中止
while(1)
{ disp();
k=getkey();//读取按键状况
if(state==0x02)//假如当时正在闹铃
{ if(k!=0xff)//只需有键按下
{ state=0x00; //退出闹铃方法,回到正常走时状况
flash=0x00; //正常显现,不闪耀
disp_on=0xf0;
}
}
else if (k==KSET_LONG)//假如长按SET键
{
set_time(ala);//设置报警时刻
}
else if(k==KSET)//假如是SET键
{ TR0=0;//封闭守时器T0
set_time(time);//设置当时时刻
TR0=1;//从头敞开守时器
}
if(update==1)
{ update=0;
hexbcd(time);
}
}
}
来历;21ic