51单片机有两个定时器 T0和T1,为16位加法计数器,由低8位TLX和高8位THX两个寄存器组成,最大计数值为65535个计数脉冲。
该计数器的计数脉冲来源有2个:
<1>系统时钟振荡器输出的12分频;
<2>T0或T1引脚输入的外部脉冲信号;
每接收到一个计数脉冲,计数器就会+1,当数值累计至全为1时(8位255,13位8191,16位65535),再输入一个计数脉冲,计数器便会溢出归零,并且计数器的溢出时TCON寄存器或者TF0或TF1位置1,同时向内核提出中断请求,如果定时/计数器工作与于定时模式,则表示隔定时时间到,如果工作于计数模式,则表示计数值已满。
假设单片机的外部晶振位12MHZ,经过12分频后输入计数器的计数脉冲位1MHZ,即每个脉冲的周期位1us,因此定时器T0的16位工作模式最大定时时间位65535us,65.5ms,如果要定时10ms的话,计数器就不能从0开始计数了。必须给它一个数值。
要定时10ms,则相当于计数10000个脉冲后计数器的值就到达65535了,那么开始计数的这个地方就是计数初值。
则为65535-10000=55535
把这个计算结果的初值写入TH0和TL0寄存器即可:
TH0 = (65535 - 10000)/256;
TL0 = (65535 - 10000)%256;
定时/计数器相关的寄存器:
<1>TMOD模式控制寄存器,不能进行复位,只能字节操作。
<2>TCON中断标志寄存器。(仍需要设置EA,IE0等等)
定时/计数器的编程思路:
两个函数:初始化函数和中断服务函数。
在初始化函数中:
<1>配置工作模式,即对TMOD寄存器编程。
<2>计算计数初值,即对THx和TLx寄存器进行赋值
<3>使能定时/计数器中断,即EX0或ET1置1
<4>打开总中断,即EA = 1
<5>启动定时器,即TR0或TR1置1
void Timer1()
{
TMOD = 0x01;
TH0 = (65535 - 10000) / 256;
TL0 = (65535- 10000) % 256;
ET0 = 1;
EA = 1;
TR0 = 1;
}
在中断服务函数中:(interrupt 1 为 定时器)
<1>重新计算THx和TLx
<2>进行间隔定时到达的逻辑编程(越少越好)
void ServiceTimer1() interrupt 1
{
TH0 = (65535 - 10000) / 256;
TL0 = (65535 - 10000) % 256;
number ++;
}
如下代码可以实现两位数码管按秒显示时间
#include "reg52.h"
unsigned char code io[10] =
{0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,
0x80,0x90};
sbit L1 = P0^0;
void HC573(unsigned char channel)
{
switch(channel)
{
case 4:
P2 = (P2 & 0x1f) | 0x80;
break;
case 5:
P2 = (P2 & 0x1f) | 0xa0;
break;
case 6:
P2 = (P2 & 0x1f) | 0xc0;
break;
case 7:
P2 = (P2 & 0x1f) | 0xe0;
break;
}
}
unsigned char number = 0;
unsigned char shuzi = 0;
void smg()
{
HC573(6);
P0 = 0x01<<6;
HC573(7);
P0 = io[shuzi/10];
}
void smg2()
{
HC573(6);
P0 = 0x01 <<7;
HC573(7);
P0 = io[shuzi%10];
}
void ServiceTimer1() interrupt 1
{
TH0 = (65535 - 10000) / 256;
TL0 = (65535 - 10000) % 256;
number ++;
}
void Timer1()
{
TMOD = 0x01;
TH0 = (65535 - 10000) / 256;
TL0 = (65535- 10000) % 256;
ET0 = 1;
EA = 1;
TR0 = 1;
}
void delay(unsigned int t)
{
while(t--);
}
void main()
{
Timer1();
while(1)
{
if(number == 100)
{
shuzi++;
number = 0;
}
smg();
delay(600);
smg2();
delay(600);
}
}