单片机驱动发光二极管数码管时钟程序

来源:本站
导读:目前正在解读《单片机驱动发光二极管数码管时钟程序》的相关信息,《单片机驱动发光二极管数码管时钟程序》是由用户自行发布的知识型内容!下面请观看由(电工技术网 - www.9ddd.net)用户发布《单片机驱动发光二极管数码管时钟程序》的详细说明。
简介:本文主要是单片机驱动发光二极管数码管时钟程序,希望对你的学习有所帮助。

单片机驱动发光二极管数码管时钟程序

​用发光二极管自己制作了一个数码管,技术的含量不是很高,但是费功夫,单片机程序是在书上看到的,作了一些小修改。硬件中没有用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);

}

}

}

提醒:《单片机驱动发光二极管数码管时钟程序》最后刷新时间 2024-03-14 00:53:35,本站为公益型个人网站,仅供个人学习和记录信息,不进行任何商业性质的盈利。如果内容、图片资源失效或内容涉及侵权,请反馈至,我们会及时处理。本站只保证内容的可读性,无法保证真实性,《单片机驱动发光二极管数码管时钟程序》该内容的真实性请自行鉴别。