题目背景
利用51单片机的定时/计数器T0的模式1实现间隔定时,每隔1秒L1指示灯闪烁一下,也就是亮0.5秒,熄灭0.5秒;每隔10秒L8指示灯闪烁一下,也就是也就是亮5秒,熄灭5秒。
分析
计数初值。
定时/计数器,本质上就是计数器,计数到溢出时产生中断。作为计数器时,计数信号来源是周期性的内部时钟脉冲。作为定时器时,计数信号来源是非周期性的外部输入信号。
51单片机一共有两个定时/计数器,T0和T1,均为16位的加法计数器。
高8位,THx,低8位,TLx。
1s = 1000ms
1ms = 1000us
1us = 1ns
51单片机内部采用12分频,如果采用12MHz的晶振,时钟脉冲是1us,最大定时时间为65535us ~ 65.5ms。
如果要定时10ms,需要给一个计数初值65535 - 10000 = 55535 = 0xd8ef,TH设置为0xd8,TL设置为0xef。也就是从55535开始计数,计数到溢出时产生中断信号,此时为10ms。
相关寄存器
除了TH和TL之外,还有TMOD寄存器和TCON寄存器。
注意TMOD只能字节寻址。
这里只需设置TMOD为0x01即可。
这里用到了TR0,来启动T0。
此外,还需要设置IE寄存器,因为和中断相关。
代码
#include <reg52.h>
#define uint unsigned int
#define uchar unsigned char// 用到的两个LED灯
sbit L1 = P0 ^ 0;
sbit L8 = P0 ^ 7;// 控制HC138译码器选通信道
void HC138(uchar 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;}
}void InitTimer0()
{TMOD = 0x01;TH0 = (65535 - 50000) / 256;TL0 = (65535 - 50000) % 256;ET0 = 1;EA = 1;// 由TR0来启动TR0 = 1;
}uchar count = 0;
void ServiceTImer0() interrupt 1
{// 没有自动重装功能,需要重新写入计数初值TH0 = (65535 - 50000) / 256;TL0 = (65535 - 50000) % 256;count ++ ;if (count % 10 == 0){L1 = ~L1;}if (count == 100){L8 = ~L8;count = 0;}
}void main()
{HC138(4);InitTimer0();while(1){}
}