【C51】单片机定时器介绍

时间:2020-12-06 23:35:48
 
简介
 
C51中的定时器和计数器是同一个硬件电路支持的,通过寄存器配置不同,就可以将他当做定时器或者计数器使用。
确切的说,定时器和计数器区别是致使他们背后的计数存储器加1的信号不同。当配置为定时器使用时,每经过1个机器周期,计数存储器的值就加1。而当配置为计数器时,每来一个负跳变信号(信号从P3.4 或者P3.5引脚输入),就加1,以此达到计数的目的。
 
标准C51有2个 定时器/计数器:T0和T1。他们的使用方法一致。C52相比C51多了一个T2。
  【C51】单片机定时器介绍
 
 
时钟周期与机器周期
 
定时器的本质原理就是:每经过1个机器周期,计数存储器的值就加1。因此当使用定时器时,就必须掌握时钟周期和机器周期的关系。
 
时钟周期 :晶振频率的倒数。如果使用的是11.0592M的晶振,那么就是 1 / (11.0592x10^6) 秒   
注:1MHz = 10^6Hz
 
 
机器周期 :标准51下,机器周期 =12倍的时钟周期。即:12 / (11.0592x10^6) 秒 。 有的增强51单片机,1个机器周期等于4倍的时钟周期,还有的更短。
 
 
 
计数存储寄存器THx&TLx
 
定时器和计数器工作,都依赖于 计数。计数则是由计数存储器THx和TLx这2个8位寄存器完成的。
对于计数器,每来一个负跳变信号(信号从P3.4 或者P3.5引脚输入),就加1,以此达到计数的目的。
对于定时器,每隔1个机器周期 加 1,假如(只是假如)一个机器周期为 1ms , 当加到1000次时,我们就认为经过了1s,这就是定时器定时依据。
 
 
T0和T1都拥有一对8bit计数存储寄存器。他们的复位值都是0。
T0 对应:TH0 ,TL0
T1 对应 : TH1 , TL1
 
sfr TL0 = 0x8A; // TL中的L是LOW的意思,代表低位,同理H代表HIGH高位。2个8位组合起来就形成了一个16位的计数器。当然也可以配置为仅仅当做8bit计数存储器用。
sfr TL1 = 0x8B;
sfr TH0 = 0x8C;
sfr TH1 = 0x8D;
 
当计数器加满后,再加1,就溢出了,溢出后自动归0。且溢出时,溢出标识位TFx 就会自动变为1(T0的溢出标志位TF0,T1的溢出标志位TF1)。如果启用了对应的中断,单片机会调用中断处理函数。
 
 
 
若TH0 和 TL0 以 16位 模式工作,那它的计数范围为   [0 , 65535 ]  ,  也就是累加 65536次发生溢出。 每累加一次是  12 / (11.0592x10^6) 秒。
那么从 0 累加到溢出 历时  ≈ 0.071s = 71ms 。
可以延时 10的整数倍ms,这样就避免了误差,以便用倍数控制更长的延时时间。所以,我么要给 TH0 和  TL0赋一个初始值,使他们的溢出周期(TH0,TL0从初始值到溢出所用的时间)减少到 10ms。
就像一个瓶子,开始装了2/3,再来就只能装1/3就溢出了。通过比例式计算:
12 / (11.0592x10^6) s       -----     1   次
10x 10-3   s                        ------          x  次         (求出 x = 9216次 ,计数9216次后溢出)
65536 - 9216 = 56320  =  二进制( 11011100   00000000)
也就是  TH0 = 11011100 , TL0 = 00000000
 
 
 
 
工作模式寄存器TMOD
 
通过TMOD来配置T0和T1的工作模式。
注意,TMOD寄存器不可位寻址(例如sbit led = P0^0 就是P0寄存器位寻址的例子),因此对它的配置需要对这个8bit寄存器整体赋值。
注:51中有些特殊功能寄存器不支持位寻址。只有寄存器的地址值能够被8整除的(即以数字0或者8结尾的地址,如0xA8, 0xD0),才能支持位寻址。不支持位寻址的,只能整体赋值。
 
 
小技巧:在对寄存器整体赋值时,要注意只修改我们想修改的位而不影响其它无关位的值,避免影响了之前对这个寄存器的配置。
 
TMOD |= 0x01;   //仅仅修改TMOD的最低位,其他位保持不变。
 
  【C51】单片机定时器介绍

 

 
 
C/T:计数器,定时器功能选择位。 1为计数器模式, 0 为定时器模式。
 
M0和M1:
 
M1
M0
模式
1
THx和TLx  组成一个16位计数存储器
1
8位重装模式。THx的值不变,负责在每次溢出后初始化TLx,仅仅由TLx计数
1
1
禁用定时器 1,定时器 0 变成 2 个 8 位定时器。很少使用。
兼容 8048 单片机的 13 位定时器,THn的 8 位和 TLn 的 5 位组
成一个 13 位定时器。很少使用。
 
GATE:门控位。
 
【C51】单片机定时器介绍

 

 
解释说明:
②处 C/T = 0 表示为定时器模式,触发信号为①处的单片机内部时钟信号。(若②处CT = 1,则触发信号为Tn脚)
③处表明,信号能触发使加法计数器加1,还得受④处控制。不然时钟信号是不能让加法计数器累加的。 ④处这个是与门,所以TRx(TR0和TR1)必须为1,表明我们要开启定时器。同时当,GATE为0,通过非门后为1,再通过或门,也是1,那么就让③处控制起来了。 若GATE为1,那么,定时器的启动停止受 TRx和 INTx 共同控制。 INTx脚为1且TRx为1才能启动定时器/计数器。
于是在一般情况下使用定时器,我们需要如下配置:
 
TRx    为  1
GATE  为 0
INTx   任意
 
 
控制寄存器TCON
 
控制寄存器就是用来控制定时器/计数器 启动和停止的,以及溢出标志位的查询和修改。TFx是计数存储器溢出标志位,只要一溢出,就马上置为1。
 
【C51】单片机定时器介绍

 

TF1:定时器/计数器1的溢出标志位。1表示计数存储器溢出,0表示计数存储器正常计数。
        清0方式:①通过代码修改TF1为0
                       ②当通过中断机制来使用定时器/计数器1时,进入中断处理函数后自动归0
TR1:定时器/计数器1的启动和停止位。1表示启动,0表示停止。
 
 
 
TF0:定时器/计数器0的溢出标志位。1表示计数存储器溢出,0表示计数存储器正常计数。
        清0方式:①通过代码修改TF0为0
                       ②当通过中断机制来使用定时器/计数器0时,进入中断处理函数后自动归0
TR0:定时器/计数器0的启动和停止位。1表示启动,0表示停止。
 
 
低4位与外部中断INT0和INT1有关,与定时器/计数器无关。这里不做介绍。
 
 
 
 
 
查询法使用T0作为定时器
 
程序1:P0_0连接驱动的LED小灯,用T0作为16位定时器,完成间隔为1s 的 blink程序。
 
#include<REGX51.H>
#include"binary.h"
#include"int51.h"
/******************************/
void timer0_init(void);
void timer0_delay(uint16_t dly);
// P0_0驱动LED小灯
#define LEDpin P0_0
void main(void)
{
    LEDpin = 0;
    timer0_init();
    TR0 = 1;
    
    for(;;)
    {
       LEDpin = 1;
       timer0_delay(1000);         //延时1000ms
       LEDpin = 0;
       timer0_delay(1000);        //延时1000ms
    }
}
/*************************
T0作为定时器的初始化
**************************/
void timer0_init(void)
{
    TMOD |= B0000_0001;        //定时器0,16位存储计数器模式
    
    TH0 = B1101_1100;       //TH0 TL0 形成数是 56320  。这样,一次溢出代表经过10ms     
    TL0 = B0000_0000;
}
/**********************
参数:
    dly,延时的毫秒数,只能是10的整数倍
***********************/
void timer0_delay(uint16_t dly)
{
    
    while(dly)
    {    
        if(TF0){
             TF0 = 0;
             TH0 = B1101_1100;  
             TL0 = B0000_0000;
             dly -= 10;             //溢出一次代表10ms
        }
    }
}
 
 
程序2:通过T0定时器的8位重装模式,使得P0_0输出PWM信号,LED为呼吸灯效果。
 
#include<REGX51.H>
#include"binary.h"
#include"int51.h"
/******************************/
void timer0_init(void);
void timer0_delay(uint16_t dly);
void pwm_duty(uint16_t d);
// P0_0驱动LED小灯
#define LEDpin P0_0
void main(void)
{    
    uint16_t i;
    LEDpin = 0;
    timer0_init();
    TR0 = 1;
    
    
    for(;;)
    {
       for(i=0;i<=500;++i)
           pwm_duty(i);
       for(i=500;i>0;--i)
           pwm_duty(i);
    }
}
/*************************
T0作为定时器的初始化
**************************/
void timer0_init(void)
{
    TMOD |= B0000_0010;      //定时器0,8位重装模式
    
    TH0 = 250;           //一次溢出代表经过6.51us     
    TL0 = 250;
}
/**********************
一次溢出代表经过6.51us
参数 c乘以6.51us 就是这个函数延时的时间   
***********************/
void timer0_delay(uint16_t c)
{
    while(c)
    {    
        if(TF0){
             TF0 = 0;   //因为是自动重装,因此不用给计数存储器赋值。
             --c;             
        }
    }
}
/*************
参数d 的范围是[0,500]。d / 500 即为 pwm输出的占空比,控制LED灯的暗亮程度
**************/
void pwm_duty(uint16_t d)
{
    LEDpin = 1;
    timer0_delay(d);
    LEDpin = 0;
    timer0_delay(500-d);
}
 
 
注:如果需要 精确的延时,使用8位自动重装模式最好,因为硬件装值(赋值给TH0)比软件装值快。但8位自动重装模式不宜做单次长时间延迟。毕竟溢出周期短。长时间延迟需要多个溢出周期,也挺消耗资源的。
 
尽量让溢出周期 越长越好。溢出周期为10ms 的优于 1ms 的。因为,在同样的延时时间下,如100ms,溢出周期为10ms 的 只需要溢出10次,为TH0 和 TL0重新赋值10次,而溢出周期为1ms的要溢出100次,为TH0 和 TL0重新赋值100次。减少溢出次数和赋值次数,可以减轻单片机的负担,提高定时的准确性。