什么是定时器?
定时器顾名思义就是用来定时的。在单片机应用中常常用于各种各样的定时。比如让LED灯每隔 1S 亮一次。 这个1S 就是由定时器做到的。
指令周期
指令周期就是单片机执行一个指令所花费的时间。这也是定时器定时的最小时间单位。时钟频率/4=指令频率。1/指令频率=指令周期。
假设现在的时钟是4MHZ ,4MHz的时钟经过4分频后变成了 1MHz 其周期为0.0000001s也就是1us,这个1us就是指令周期,这1us也就是定时器定时的最小单位。
定时器与预分频器
假设在没有预分频器情况下。开启定时器 每隔一个指令周期定时器就加一。假设时钟是4MHz 也就是每隔 1us 定时器加一。
如果有了预分频器假设预分频器设置成2分频,定时器就 每隔2个指令周期定时器加一。如果预分频器设置成4分频,定时器就 每隔4个指令周期定时器加一,以此类推。
定时器中断标志位
如: TMR0 这个是8位的定时器,也就是8位的寄存器。8位的寄存器能代表的数值为0~255.也就是说定时器可以从0开始加一直加到255.到255后再加一就又变成0。此时TMR0定时器中断标志位 (TMR0IF)变成 1.(如果中断没有开启,并不执行中断程序。)
到底从时钟频率一直到定时器中断溢出之间是什么关系呢?
下面我画了一个流程图我们用频率的方式来理解这一切。假设时钟频率是4MHz ,定时器预分频值为2,定时器初始值为0.
1。首先4MHz 的时钟 4分频后变成 1MHz的指令频率;
2。然后预分频器 2 分频后变成 0.5MHz的频率供给定时器;
3。定时器经过256分频后变成约1952Hz的频率溢出中断;
然后我们再用周期的方式来理解这一切。
1。首先0.25us时钟周期4分频后变成 1us指令周期;
2。然后预分频器 2 分频后变成 2us周期 供给定时器;
3。定时器每隔2us加一 ,加到256次 256X2us=512us溢出中断 ;
希望上面的流程图能帮你梳理一下概念。
实例说明:
假设时钟周期为4MHZ,每隔50MS点亮LED,每隔50MS灭掉LED。这样的程序要如何做到呢。
这50ms如何做到.
1,得到指令周期
4MHz/4=1MHz
1/1MHz=0.0000001s=1us
2,得到预分频
定时器定时的最大时间要超过这50mS,所以预分频器要选择256
预分频X256=最大的定时时间。256X256=65536us=65.536ms 大于50ms
3, 计算定时器初始值
(定时器最大值+1)- (定时时间/预分频)=定时器的初始值。
255+1=256
50000/256=195.3125
256-195.3125=60.6875 四舍五入 定时器初始值为61.
设置相关的寄存器。
OPTION_REG寄存器中我们一般需要设置三处。
PS<2:0>设置用来设置预分频预分频范围从2 ~256
PSA设置成0 讲预分频器分配给Timer0模块
TMR0CS设置成0 内部指令周期时钟。
实例程序:
/*开发环境 MPLAB X IDE 型号PIC16LF1823*/
#include
__CONFIG(FOSC_INTOSC&WDTE_OFF&PWRTE_ON&MCLRE_OFF&CP_ON&CPD_OFF
&BOREN_ON&CLKOUTEN_OFF&IESO_ON&FCMEN_ON);
__CONFIG(PLLEN_OFF&LVP_OFF) ;
#define LED LATA5/*也可用 #define LED RA5,只是PIC16LF1823 输出电平的时候,直接控制LATA5执行速度更快,因为传给RA5的数据最终也是传给LATA5才执行的*/
void init_fosc(void)
{
OSCCON= 0x68;//时钟设置为4MHz
}
void init_gpio(void)
{
PORTA = 0;
LATA = 0;
ANSELA = 0;
TRISAbits.TRISA5=0; //RA5口设置成输出 用来控制LED
}
void init_timer0(void)
{
OPTION_REG=0x87; //预分频为256
}
int main(int argc, char** argv)
{
init_fosc();
init_gpio();
init_timer0();
TMR0IF=0;//清除TMR0中断标志位
TMR0=61;//设置中断初始值61
while(1)
{
if(TMR0IF==1)//定时时间到了吗??
{
LED = ~LED;//改变LED的状态
TMR0IF=0;//清除TMR0中断标志位
TMR0=61;//设置中断初始值61为下次50ms定时做准备
}
}
}