使用定时器中断替代延时函数的方法及原理思想(附带例程)

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档

目录

前言

一、使用定时器中断的优缺点

二、使用步骤

1.原理思路

2.相关代码

(1).定时器初始化

(2).stm32f10x_it.c(该文件专门用于存放中断服务函数) 

(3).主函数main.c

 3.效果:

总结




前言

本文主要基于stm32f103系列讲解定时器中断进行计时代替延时函数,其中思路和原理同时适用于其他系列的单片机。


一、使用定时器中断的优缺点

相较于普通延时函数(delay),定时器中断计时无论是从代码的体量,还是使用,都要比延时函数更多更复杂,也更难理解。既然延时函数如此便捷,那我们又为什么不使用它呢?

首先在使用延时函数时,程序会停止在延时函数的位置,直到延时结束,在一些简单程序,使用延时函数的影响确实可以忽略不计,但是过多的延时函数会使程序变得臃肿,编译执行时间大大增加,程序的精度会下降,例如利用按键控制led灯时,led灯会不受按键控制,出现时灵时不灵的情况,这种情况就是延时函数使用过多造成的,更严重的甚至会造成单片机宕机。

所以,为了避免这种情况的发生,我们可以使用定时器中断来代替延时函数,下面是我个人的一些理解和认识,如有不足,欢迎指正。

二、使用步骤

1.原理思路

首先,我们需要配置一个定时器,stm32一般选用通用定时器即可,我们可以利用定时器初始化函数设定定时器计时一次的时间,公式如下:

定时时间=(arr+1)(psc+1)/Tclk

其中arr为自动重装载值,psc为预分频系数,TCLK为时钟频率,例如:TCLK=72MHz,那么psc=71,所以可以理解为时间就是(arr+1)微妙,那么如果我想定时1ms,arr取999即可(arr,psc为定时器初始化函数形参)。

当定时器的计数器计数到自动重装载值时,进入中断服务函数,这时我们需要设置一个标志位(flag)和作为计数用的值(count),每次进入中断count执行自加或自减,当其自加或自减到设置的数值时,标志位反转。 

例如:定时200ms,已知1ms进入一次中断,标志位flag初始值为0,那么让count自加到200,因为每次自加是1ms,自加两百次就是200ms,这时flag置1表示到达200毫秒并可以执行相关程序。

2.相关代码

(1).定时器初始化

(这里我初始的是TIM2)

TIMER.H

#ifndef __TIMER_H
#define __TIMER_H

#include "sys.h"
#include "stm32f10x_tim.h"

void TIM2_Int_Init(u16 arr,u16 psc);//通用定时器的初始化函数;arr:自动重装载值psc:预分频系数


#endif

TIMER.C 

#include "TIMER.h"


void TIM2_Int_Init(u16 arr,u16 psc)//通用定时器3的初始化函数
{
	//定义相关结构体
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStrue;//定义一个定时器初始化的结构体
	NVIC_InitTypeDef NVIC_InitStrue;//定义中断优先级初始化的函数
	
	//使能定时器时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);//使能通用定时器3的时钟
	
	//设置并初始化定时器TIM2
	TIM_TimeBaseInitStrue.TIM_CounterMode=TIM_CounterMode_Up;//计数模式设置为向上计数
	TIM_TimeBaseInitStrue.TIM_Period=arr;//计数器模式为向上计数时,定时器从0开始计数,超过arr    
                                         //触发定时中断服务函数
	TIM_TimeBaseInitStrue.TIM_Prescaler=psc;//预分频系数,决定每一个计数的时长
	TIM_TimeBaseInitStrue.TIM_ClockDivision=TIM_CKD_DIV1;//一般不使用,默认为TIM_CKD_DIV1
	TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStrue);//根据参数初始化定时器TIM3

	//使能定时器中断
	TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);//使能TIM3的中断,中断模式为更新中断
	
	//初始化定时器中断,定时器中断优先级设置
	NVIC_InitStrue.NVIC_IRQChannel=TIM2_IRQn;//中断通道设置为TIM3
	NVIC_InitStrue.NVIC_IRQChannelCmd=ENABLE;//使能中断
	NVIC_InitStrue.NVIC_IRQChannelPreemptionPriority=1;//抢占优先级为1级
	NVIC_InitStrue.NVIC_IRQChannelSubPriority=1;//响应优先级为1级
	NVIC_Init(&NVIC_InitStrue);//根据参数初始化中断寄存器
	
	//使能定时器
	TIM_Cmd(TIM2,ENABLE);//使能通用定时器TIM2
}

(2).stm32f10x_it.c(该文件专门用于存放中断服务函数) 


#include "stm32f10x_it.h"


extern	u8 flag;//标志位

void TIM2_IRQHandler()
{
	static u8 count;


  if(TIM_GetITStatus(TIM2, TIM_IT_Update)==1) //当发生中断时状态寄存器(TIMx_SR)的bit0会被硬件 
                                              //置1
	{
		if(count--==0)
		{
			count=20;
			flag=1;
		}

		TIM_ClearITPendingBit(TIM2, TIM_IT_Update); //状态寄存器(TIMx_SR)的bit0置0
	}
}

(3).主函数main.c

#include "stm32f10x.h"
#include "stm32f10x_conf.h"
#include "sys.h"

u8 flag=0;


int main(void)
{
	delay_init();					//初始化延时函数
	LED_Init();						//初始化led灯
	TIM2_Int_Init(9999,71);			//定时10ms进入一次中断

    while(1)
    {
            if(flag==1)    //标志位置一代表一次定时完成
			{
				PAout(2)=~PAout(2);    //led2取反
				flag=0;                //标志位置0,再次计时(该程序设置一次定时为200ms)
			}
    }
}

 3.效果:

每隔200msLED灯闪烁。


总结

以上是本人对定时器中断的一些认识(本人小白),其中有些地方讲解不到位,还望各位指正,欢迎评论区留言。

物联沃分享整理
物联沃-IOTWORD物联网 » 使用定时器中断替代延时函数的方法及原理思想(附带例程)

发表评论