基于单片机的简易定时报警器电路设计
中断服务程序流程图
(4)软硬件调试方案
软件调试方案:伟福软件中,在“文件新建文件”中,新建C语言源程序文件,编写相应的程序。在“文件新建项目”的菜单中,新建项目并将C语言源程序文件包括在项目文件中。
在 “项目编译”菜单中将C源文件编译,检查语法错误及逻辑错误。在编译成功后,产生以 “*.hex”和“*.bin” 后缀的目标文件。
硬件调试方案:在设计平台中,将单片机的P3.0-P3.3分别与独立式键盘的相应位通过插线连接起来。
在伟福中将程序文件编译成目标文件后,运行MCU下载程序,选择相应的flash 数据文件,点击“编程”按钮,将程序文件下载到单片机的Flash中。
然后,上电重新启动单片机,检查所编写的程序是否达到题目的要求,是否全面完整地完成试题的内容。
3.1.3 程序设计(仅供参考的C语言源程序)
//晶振:11.0592M T1-250微秒 按键P10 P11 P12 P13
/*变量的定义:
show_val: 显示的值0-59
init_val: 初始值
state_val: 状态值 0-计数状态;1-设置状态;2-闪烁状态
shan_val:
key_val1: 四个按键的值 255-无键;1-设置键 2-增一键 3-减一键 4-确定键
T1_cnt: 定时器计数溢出数
cnt_val: 倒计时的数值
led_seg_code:数码管7段码
*/
#include "reg51.h" //包含文件
sbit P1_0=P1^0; //设置键
sbit P1_1=P1^1; //增一键
sbit P1_2=P1^2; //减一键
sbit P1_3=P1^3; //确定键
unsigned char data shan_val; //闪烁时LED的开/关状态
unsigned char data cnt_val; //保存倒计数的当前值
unsigned int data T1_cnt; //保存定时器溢出次数
unsigned char data key_val_new,key_val_old;//存放当前扫描的键和前一次按下的键值
unsigned char data state_val; //状态值
unsigned char data show_val; //存放需要在数码管显示的数字
unsigned char data init_val; //暂存倒计数的初始值
char code led_seg_code[10]={0x3f,0x06,0x05b,0x04f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
//----------延时--------------
void delay(unsigned int i) //大约延时i*2个微秒
{ while(--i);}
//-----------按键扫描-------------
unsigned char scan_key()
{ unsigned char i;
i=P10x0f;
delay(100); //延时,去抖动
if (i==(P10x0f))
{ if (P1_0==0)
{ i=1; }
else
{ if (P1_1==0)
{ i=2;}
else
{ if (P1_2==0)
{ i=3;}
else
{ if (P1_3==0)
{ i=4;} }
} } }
else
{ i=255; }
return i;
}
//---------数码管显示---------------
void led_show(unsigned int v)
{
unsigned char i;
if (state_val!=2) //动态扫描
{i=v%10; //取要显示的数的个位
P0=led_seg_code[i]; //转换为7段码
P2=0xfe; //显示个位
delay(15); //延时
i=v%100/10; //取十位
P0=led_seg_code[i]; //转换为7段码
P2=0xfd; //显示十位
delay(5); //延时
}
else
{ P0=led_seg_code[0]; //处于闪烁状态
if (shan_val)
{ P2=0xff; } //将数码管的关闭
else
{ P2=0xfc; } //将数码管的打开
}
}
//----------定时器T1中断服务程序---------------
void timer1() interrupt 3 //T1中断,250us中断一次
{ T1_cnt++;
switch (state_val)
{ case 0:
if(T1_cnt>3999) //如果计数>3999, 计时1s
{ T1_cnt=0;
if(cnt_val!=0)
{ cnt_val--;}
else
{state_val=2;} //定时计数到0时,切换状态
show_val=cnt_val;
}
break;
case 2:
if(T1_cnt>1999) //如果计数>1999, 计时0.5s
{ T1_cnt=0; shan_val=!shan_val; } //闪烁状态
break;
}
}
//---------主程序----------------
main()
{init_val=59; //初始化各变量
cnt_val=init_val;
show_val=cnt_val;
state_val=0;
key_val_old=255;
T1_cnt=0;
shan_val=0; //初始化51的寄存器
TMOD=0x20; //用T1计时 8位自动装载定时模式
TH1=0x19; //250微秒溢出一次; 250=(256-x)*12/11.0592 -> x= 230.4
TL1=0x19;
EA=1; //打开总中断允许
ET1=1; //开中断允许
TR1=1; //开定时器T1
while(1)
{ key_val_new=scan_key(); // 255表示无键按下
if (key_val_new!=key_val_old)
{ // 只有当前扫描的键值与上次扫描的不同,才判断是有键按下
key_val_old=key_val_new;
switch (key_val_new)
{ case 1: //设置键
state_val=1; //处于设置状态
TR1=1; //停止计时
show_val=init_val; //显示原来的倒计数初始值
break;
case 2: if(state_val==1) //只有在设置状态,增1键才有用
{ if (init_val>0) //更改原来的倒计数初始值
{init_val--; }
else
{init_val=59;}
show_val=init_val;//显示更改后的倒计数初始值
}
break;
case 3: if(state_val==1) //只有在设置状态,减1键才有用
{ if (init_val59) //更改原来的倒计数初始值
{init_val++; }
else
{init_val=0;}
show_val=init_val; //显示更改后的计数初始值
}
break;
case 4: if(state_val!=0) //如果已处于计数模式,确认键不起作用
{ cnt_val=init_val; //将初始值赋给计数变量
show_val=cnt_val; //将计数变量的数字显示
TR1=1; //启动定时器T1
state_val=0; //将状态切换为计数模式
}
break;
}
}
led_show(show_val); //动态扫描
}
}
- FPGA的DSP性能揭秘(06-16)
- 基于单片机通用引脚的软件UART设计(10-16)
- 分时操作系统思想在单片机中的具体应用 (10-30)
- 基于AT89C51+DSP的双CPU伺服运动控制器的研究(05-26)
- 关于RTX51 TINY的分析与探讨(05-30)
- 基于MC9S12DGl28单片机的智能寻迹车设计(04-03)