电工优优今天要和大家分享的单片机驱动发光二极管数码管时钟程序相关信息,接下来我将从单片机驱动发光二极管,单片机驱动发光二极管电路图,单片机驱动发光二极管的方式这几个方面来介绍。
单片机驱动发光二极管数码管时钟程序
用发光二极管自己制作了一个数码管,技术的含量不是很高,但是费功夫,单片机程序是在书上看到的,作了一些小修改。硬件中没有用DS1302,单片机晶振用12M(程序以12M晶振进行计算的),走时比用DS1302还要准。”特此收集转载。
接口
数码管:P0^0~P0^7
位选: P2^0~P2^3
设置键 P1^5~P1^7(短按:设置显示时间,长按:设置闹钟时间)
加1键 P1^6
减1键 P1^7
Beep P3^7
//作者:whw8099
#include
#include
#define uchar unsigned char
#define uint unsigned int
#define KSET 0x60 //设置当前时间键
#define KSET_LONG 0x61 //设置闹铃时间键
#define KINC 0x50 //加1键
#define KDEC 0x30 //减1键
sbit SPK=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位显示缓存
uint count; //用于累加定时器T0的溢出中断次数,取值为0到4000
uchar time[3]={0,0,0}; //当前时、分、秒
uchar ala[2]={0,0}; //报警时、分
uchar state; //state=0:正常走时方式;
//state=1:设置方式;
//state=2:正在闹铃
bit update; //时间更改标志
uchar flash=00; //闪烁控制字
uchar disp_on=0xf0; //显示控制位
void delay(void)
{ uchar i;
for(i=0;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 uchar cn=0;
uchar n,bsel;
bsel=0xfe; //最初点亮最低位
for(n=0;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;
}
}
void time0(void) interrupt 1 using 1
{ static uint n=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
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)
{ uchar key;
uchar t;
disp();
if((key=P1&0x70)==0x70) return 0xff;
for(t=0;t
if((key=P1&0x70)==0x70) return 0xff;
while((P1&0x70)!=0x70) //检测按键时间
{ disp();
if(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; //当前状态恢复为正常走时状态
}
void main(void)
{
uchar k;
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);
}
}
}
以上就是"电工优优"为大家介绍的单片机驱动发光二极管的相关信息,想了解更多"单片机驱动发光二极管数码管时钟程序,单片机驱动发光二极管,单片机驱动发光二极管电路图,单片机驱动发光二极管的方式"相关知识,请收藏电工无忧吧。