
/**** 本程序中,晶振为12MHz, ****/
/**** 时间控制采用定时中断控制方式。 ****/
/**** 模式和时间调整采用查询方式。 ****/
#include<reg52.h>
sbit P20=P2^0 //分个位控制端
sbit P21=P2^1 //分十位控制端
sbit P22=P2^2 //时个位控制端
sbit P23=P2^3 //时十位控制端
sbit led=P2^7 //second display led
sbit key0=P3^0 //模式设置
sbit key1=P3^1 //加
sbit key2=P3^2 //减
unsigned char hour,min,sec,T50ms
unsigned char modstate //模式状态
unsigned char code table[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xff}//段码
void init() //初始化子程序声明
void delay500us(unsigned char X) //延时子程序声明
void display() //显示子程序声明
void display001() //显示子程序声明
void display002() //显示子程序声明
void keyscan() //按键识别子程序声明
void main()
{
init()
while(1)
{
keyscan()
}
}
void init() //初始化子程序
{
TMOD=0x01
TH0=(65536-49990)/256
TL0=(65536-49990)%256
ET0=1
EA=1
TR0=1
}
void delay500us(unsigned char X)
{
unsigned char i,j
for(i=Xi>0i--)
for(j=248j>0j--)
}
void timer0() interrupt 1 //timer0中断服务子程序,定时时间为50ms,本程序加了10us的时间修正量
{
TMOD=0x01
TH0=(65536-49990)/256
TL0=(65536-49990)%256
T50ms++
if(T50ms>=20)
{
T50ms=0
sec++
if(sec>=60)
{
sec=0
min++
if(min>=60)
{
min=0
hour++
if(hour>=24)hour=0
}
}
}
}
void display()
{
P20=1
P21=1
P22=1
P23=1
P0=table[hour/10]
P23=0
delay500us(5)
P20=1
P21=1
P22=1
P23=1
P0=table[hour%10]
P22=0
delay500us(5)
P20=1
P21=1
P22=1
P23=1
P0=table[min/10]
P21=0
delay500us(5)
P20=1
P21=1
P22=1
P23=1
P0=table[min%10]
P20=0
delay500us(5)
if(T50ms<=10)led=0
if(T50ms>10)led=1
}
void display001()
{
P20=1
P21=1
P22=1
P23=1
P0=table[hour/10]
P23=0
delay500us(10)
P20=1
P21=1
P22=1
P23=1
P0=table[hour%10]
P22=0
delay500us(10)
}
void display002()
{
P20=1
P21=1
P22=1
P23=1
P0=table[min/10]
P21=0
delay500us(10)
P20=1
P21=1
P22=1
P23=1
P0=table[min%10]
P20=0
delay500us(10)
}
void keyscan() //按键识别钟程序
{
while(modstate==0)
{
display()
if(key0==0)
{
display()
if(key0==0)modstate++ //这两句加在一起为延时10ms软件防抖设计。
while(key0==0)display001() //等待按键释放。
}
}
//****************************************************************************//
while(modstate==1)
{
display001()
if(key0==0)
{
display001()
if(key0==0)modstate++ //这两句加在一起为延时10ms软件防抖设计。
while(key0==0)display002() //等待按键释放。
}
if(key1==0)
{
display001()
if(key1==0)
{
hour++
if(hour>=24)hour=0
while(key1==0)display001()
}
}
if(key2==0)
{
display001()
if(key2==0)
{
hour--
if(hour>=24)hour=0
while(key2==0)display001()
}
}
}
//****************************************************************************//
while(modstate==2)
{
display002()
if(key0==0)
{
display002()
if(key0==0)modstate=0 //这两句加在一起为延时10ms软件防抖设计。
while(key0==0)display() //等待按键释放。
}
if(key1==0)
{
display002()
if(key1==0)
{
min++
if(min>=60)min=0
while(key1==0)display002()
}
}
if(key2==0)
{
display002()
if(key2==0)
{
min--
if(min>=60)min=0
while(key2==0)display002()
}
}
}
}
单片机程序是基于硬件基础上开发的,你至少添加注释和硬件环境说明,不然大家都不知道怎么帮你。你直接给出代码,也不清楚需要指正什么,这里也没有设置时间/闹钟的代码,发文前应该清楚表达要问的内容。
1)假设你的晶振是12MHz,那一个机器周期是1MHz,定时器的最长定时周期是(65535/10^6)你想要1/6秒触发一次中断就必须结合软件计时,为了尽量精确:硬件计时40000次,软件计时25次那么定时器的设这应该是TH1=(65535-40000)%256;TL1=(65535-40000)%256,(注意计数器大部分是加法计数)
2)按键监听没有给出思路,也没有处理按键抖动,我只能分析程序:
有一个外部中断触发时间设置,P34设置小时、P35设置分钟,但是调整时间后没有进行显示,你至少要让数码管显示结果才能知道怎么调整。
#define uchar unsigned char // (先定义一下方便使用)
#define uint unsigned int
#define ulong unsigned long
#include <reg52.h>//包括一个52标准内核的头文件
uchar code zixing[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90}
uchar code ziwei[]={0x10,0x20,0x40,0x80}
uchar *h,*m
uchar hour=12,min=0,sec=0,h_ding,m_ding,count=0,ms_50,ms_10,miaobiao_sec
sbit dp=P0^7
sbit k1=P3^2
sbit k2=P3^3
sbit k3=P3^4
sbit sound=P2^3
sfr ZIXING=0x80//p0口输出字形
sfr ZIWEI=0xa0//P2口输出字位
typedef enum //枚举类型:4种状态//
{
CLOCK,
SET_RING1,
SET_RING2,
MIAOBIAO_RUN,
}STATE
STATE status
typedef struct //定时时间的数据结构//
{
uchar hour
uchar min
}DINGSHI_TIME
DINGSHI_TIME dingshi_time[2]
#define NO_KEY -1 //定义一下方便使用
#define KEY_1 1
#define KEY_2 2
#define KEY_3 3
bit show_on_flag
//#######定时器T0中断服务程序########//
Timer0() interrupt 1
{
TH0=0x3cTL0=0xb4//50ms
ms_50++
if(ms_50==20) {sec++ms_50=0}
if(sec==60) {sec=0min++}
if(min==60) {min=0hour++}
if(hour==24) {hour=0}
}
//#######定时器T1中断服务程序########//
Timer1() interrupt 3
{
TH1=0xd8TL1=0xf4//10ms
ms_10++
if(ms_10==100)
{
miaobiao_sec++
ms_10=0
}
if(miaobiao_sec==60)
miaobiao_sec=0
}
//##############数码管显示#####################//
show(uchar time_high,uchar time_low)
{
uchar m
uint n,k
k=time_high*1000/10+time_low
for(m=0m<4m++)
{
ZIXING=zixing[k%10]
if(ms_50<10)
dp=0
ZIWEI=~ziwei[m]
for(n=0n<500n++)
ZIXING=0xff
for(n=0n<100n++)
k=k/10
}
}
//显示 on 定时编号//
show_on(uchar id) //id:定时编号//
{
uchar m
uint n
for(m=0m<2m++)
{
ZIXING=zixing[id%10]
dp=0
ZIWEI=~ziwei[m]
for(n=0n<500n++)
ZIXING=0xff
for(n=0n<80n++)
id=id/10
}
ZIXING=0xc8//字母n//
ZIWEI=~ziwei[2]
for(n=0n<500n++)
ZIXING=0xff
for(n=0n<200n++)
ZIXING=0xc0//字母0//
ZIWEI=~ziwei[3]
for(n=0n<500n++)
ZIXING=0xff
for(n=0n<200n++)
}
/*************启动秒表***************/
void run_miaobiao()
{
ms_10=0
miaobiao_sec=0
TH1=0xd8//标准是0xd8e4
TL1=0xf4
// TR1=1//启动定时器1
}
显示秒表:
void show_miaobiao()
{
show(miaobiao_sec,ms_10)
}
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)