1、实验目的
(1)使用定时器来完成LED闪烁
(2)原来实现闪烁时中间的延迟是用delay函数实现的,在delay的过程中CPU要一直耗在这里不能去做别的事情。这是之前的缺点
(3)本节用定时器来定一个时间(譬如0.3s),在这个定时器定时时间内CPU还可以去做主任务,定时时间到产生中断,在中断处理程序isr中让LED闪烁即可。
如何编程:
(1)定时(timer初始化)
(2)主程序该干嘛干嘛
(3)中断处理程序
我们开发板的定时器最多能订多长时间?
内部时钟频率是1MHz,时钟周期是1us。最多能定65535(16位定时器),也就是说最大定时时间为65535*1us=65535us=65.535ms。
如果要定比较长的时间(譬如2s),定时器直接是不能满足的,解决办法是多次定时后加起来构成一个长时间。
我们使用定时器T0工作时
计算TL0和TH0:
(1)确定自己定时时间,定为50ms
(2)确定内部时钟周期,因为12T模式,外部晶振12MHz,所以为:1us
(3)定时个数就是:50ms/1us = 50000
(4)TL0 = 50000 % 256, TH0 = 50000 / 256
#include <REGX51.H>
#define SMG P0
sbit LED = P1^0;
unsigned char count;
/****定时到时执行的中断程序****/
void DingShi() interrupt 1
{
/*每一次定时到后都要重置脉冲个数,以便进行第二次定时*/
TL0 = 175; //低8位为(0x50)0101 0000的反码
TH0 = 60; //高8位为(0xc3)1100 0011的反码
if(count-- == 0) //这里相当于50ms*20=1000ms(1s)
{
count = 20;
LED = !LED;
}
}
void delay(void)//延迟函数
{
unsigned char a,b;
for(a=200 ;a>0 ;a--)
for(b=200 ;b>0 ;b--)
;
}
void main(void)
{
unsigned char i;
TMOD = 0X01; //0000 0001 使用的是T0定时器,工作模式是16位定时器
/****设置脉冲个数位50000个,既为50ms****/
TL0 = 175; //低8位为(0x50)0101 0000的反码
TH0 = 60; //高8位为(0xc3)1100 0011的反码
TR0 = 1; //计数器,开始计数
ET0 = 1; //开启T0中断
EA = 1; //开启中断总开关
count = 20;
while(1)//执行主任务
{
unsigned char val[16] = {0xc0, 0xf9, 0xa4, 0xb0,
0x99, 0x92, 0x82, 0xf8,
0x80, 0x90, 0x88, 0x83,
0xc6, 0xa1, 0x86, 0x8e};//0到F的数码表
for(i=0 ;i<=15 ;i++)
{
SMG = val[i];
delay();
}
}
}
上面程序结果是静态数码管不断的重复显示0到F,而LED1每隔1秒亮灭一次。