微波EDA网,见证研发工程师的成长!
首页 > 硬件设计 > 嵌入式设计 > 51单片机小闹钟

51单片机小闹钟

时间:11-25 来源:互联网 点击:
以前用51的时候,在Proteus里边仿真过很多东西,51的一些基本功能都做过实验,复杂点的东西虽然也有,但是都是随性而来,需要什么用什么,程序显得臃肿难读,效率低下,且没有及时把分析记录下来

有时间自己再做一遍闹钟,跟以前做的感觉差别很大,以前好像都是COPY,自己改动的很少,而且老觉得程序长,当自己去设计步骤时,出现的问题却不在这个地方,虽然51对于很多人而言,已再熟悉不过,但是我觉得算法都一样,在很熟悉的51上验证算法是很有效率的

功能:

1:时钟显示

2:时钟可调

3:可设闹钟

4:闹钟随时可关

思路:

1:时钟用定时器1控制,每隔一秒,秒变量+1,

数码管扫描显示时钟

2:设置2个可调时分按键,按键按下,对应时间变量+1

3:设置模式键,奇数次按下,改变闹钟时分

偶数次按下,改变时钟时分

不断地查询检测,时钟时分==闹钟时分?,如果等于,则相应闹铃

4:设置关闹铃键,按下时关闭闹铃

在这个地方会出现一个小问题,闹钟是否开启需要一直扫描,关闹钟按键也需要一直扫描,

当按键按下,在此刻关掉闹钟,但是,在这一分钟之内,闹钟都有开启的理由,故关了之后,

再次检测闹钟时,闹钟会再次打开,

改善方法:设置标志量,当关闹钟之后的额定时间内,不再检测闹钟,额定时间与闹钟的控制精度有关


程序如下:


#include

#define uchar unsigned char
#define uint unsigned int

sbit beep=P3^1;
sbit led=P3^0;

sbit hour_key=P3^4;//调节小时键
sbit minu_key=P3^5;//调节分钟键

sbit stopsounder_key=P3^3; //关闹钟按键

bit sounder_flag=0;

sbit mode_key=P3^7; //调节时钟还是闹铃控制按键

uchar mode_number;//模式计数

uchar sec_count;

uint a,b; //模式计数

uchar minu0=1;
uchar hour0,sec0, //闹钟

hour1,minu1,sec1,//时钟

h1,h2,m1,m2,s1,s2,//显示位

k;//状态转换标志

uchar code select[]={0x7f,0xbf,0xdf,0xef,0xf7,0xfb,0xfd,0xfe};

uchar code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};

void keyscan();
void init();
void delay(uchar z);
void display(uchar,uchar,uchar);
void mode_check();

void init()
{
a=0;
b=0;
k=0;

hour1=0;
minu1=0;
sec1=0;

hour0=0;
minu0=1;
sec0=0;

TMOD=0x11; //定时器0,1工作于方式1;赋初值

TH0=(65536-5000)/256;
TL0=(65536-5000)%6;

TH1=(65536-50000)/256;
TL1=(65536-50000)%6;


EA=1;

EX0=1;

//ET0=0;
ET1=1;

IT0=1; //P3.2引脚下经沿产生外部中断0

//PX0=1;

//TR0=1; //初始,秒表不工作

TR1=1; //时钟一开始工作

}

void delay(uchar z)
{
int x,y;
for(x=z;x>0;x--)
for(y=110;y>0;y--);
}

void ex0_int() interrupt 0
{

//作用待添加
}



void keyscan()
{

if(hour_key==0) //时按键
{

delay(10);
if(hour_key==0)
{
while(!hour_key)display(hour1,minu1,sec1); //消除数码管的抖动

if(mode_number%2==0)//如果在时钟模式下,时钟时+1
hour1++;
else hour0++;//如果在闹钟模式下,闹钟时+1

if(hour1==12)
{
hour1=0;
}

if(hour0==12)
{
hour0=0;
}

}
}

if(minu_key==0) //分按键
{

delay(10);
if(minu_key==0)
{
while(!minu_key)display(hour1,minu1,sec1); //消除数码管的抖动
if(mode_number%2==0)
minu1++;
else
minu0++;

if(minu1==60)
{
minu1=0;
}

if(minu0==60)
{
minu0=0;
}

}

}


if(mode_key==0) //模式按键

{

delay(10);
if(mode_key==0)
{

while(!mode_key)display(hour1,minu1,sec1); //数码管的抖动

mode_number++;


}

}

if(stopsounder_key==0) //关闹钟按键
{
sounder_flag=1;
delay(10);
if(stopsounder_key==0)
{

while(!stopsounder_key)display(hour1,minu1,sec1);

beep=1;
}

}


}



void display(uchar hour,uchar minu,uchar sec)
{
h1=hour/10;
h2=hour;

m1=minu/10;
m2=minu;

s1=sec/10;
s2=sec;

P2=0xff;
P1=table[h1];
P2=select[0];
delay(5);

P2=0xff;
P1=table[h2];
P2=select[1];
delay(5);

P2=0xff;
P1=0x40;;
P2=select[2];
delay(5);

P2=0xff;
P1=table[m1];
P2=select[3];
delay(5);

P2=0xff;
P1=table[m2];
P2=select[4];
delay(5);

P2=0xff;
P1=0x40;
P2=select[5];
delay(5);

P2=0xff;
P1=table[s1];
P2=select[6];
delay(5);

P2=0xff;
P1=table[s2];
P2=select[7];
delay(5);

}

void timer1_int() interrupt 3 //控制时钟工作 50ms
{
TH1=(65536-50000)/256;
TL1=(65536-50000)%6;
b++;
if(b==10)led=~led;
if(b==20)
{
led=~led;
b=0;
sec1++;
if(sec1==60)
{

sounder_flag=0; //每隔一分钟,释放检测闹钟

sec1=0;
minu1++;
if(minu1==60)
{
minu1=0;
hour1++;
if(hour1==24)
hour1=0;

}
}

}

}

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top