关于STM32定时器使用的一个注意事项(以此为前车之鉴,重要!)

    我们平时使用定时器的时候多数都是处于开启状态,平时的定时中断书写格式一般是:

void TIM3_IRQHandler(void)
{
if(TIM_GetITStatus(TIM3, TIM_IT_Update) == SET)
{

TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
//要处理的事件内容。。。。

}

}

   但是,项目的实验过程中,我使用的定时器处理事件稍微有点特殊,即,定时器不是一直处于开启状态, 而且关闭时候也是在中断里关闭。大概形式这样:

void TIM3_IRQHandler(void)
{
if(TIM_GetITStatus(TIM3, TIM_IT_Update) == SET)
{

TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
//要处理的事件内容。。。。

TIM_Cmd(TIM3, DISABLE);  //失能(函数外使能)

}

}


看似没错,而且也看似正常。但是,处理的事件内容出现了很多未知错误(由于我的这个处理事件有很强的时序性,开始和结束都比较严格),无法正常执行。通过后来的调试中发现(把处理时间改为点灯或者打印输出方式),发现是:TIM_Cmd(TIM3, DISABLE);  扰乱了时序关系。当失能后,其实中断并没有真正失能,还会再进入一次中断,因此事件又被执行了一次,对于时序比较严格的事件,就产生了问题!

找到了原因,因此,我猜测虽然定时器失能并且关闭了定时器,但是可能中断标志位并没真正清除,虽然中断开始已经清除过一次,但估计因为失能使得标志位又被置位了,因此,我在失能前面加了句清除中断更新标志位,如下:

void TIM3_IRQHandler(void)
{
if(TIM_GetITStatus(TIM3, TIM_IT_Update) == SET)
{

TIM_ClearITPendingBit(TIM3, TIM_IT_Update);
//要处理的事件内容。。。。

TIM_ClearITPendingBit(TIM3, TIM_IT_Update);//再清除标志位

TIM_Cmd(TIM3, DISABLE);  //失能(函数外使能)

}

}


果然,程序可以正常的时序运行。

比较纳闷关定时器前又得清下标志位,因此引起了另一个好奇心,是不是在其他地方关闭定时器(如主函数),也得这样做才可以。所以对这个好奇心进行了下测试。发现:如果把关闭定时器放到了主函数后,不用再清中断标志位。能正常把定时器关闭,并不会进入中断。

通过这次的问题,浪费了很多时间解决,不过也吸取到了点经验,但对于内在真正原因:在中断里失能和中断外失能效果为什么不一样,暂时还没搞清楚。。。但这个可以作为以后的前车之鉴,以及大家的前车之鉴,少走弯路。

STM32中,定时器可以被用来实现许多不同的功能。其中一个常见的应用就是测量时间间隔。在本文中,我们将介绍如何使用STM32定时器实现1秒钟的自动增减。 首先,我们需要配置定时器的基本设置。具体来说,我们需要选择一个适当的时钟源,配置定时器的预分频器,以及设置定时器的计数模式。在本例中,我们将使用定时器TIM3,并将其配置为1Hz的计时器。 接下来,我们需要编写一些代码来处理定时器中断事件。在每次定时器溢出时,我们将自动增加或减少一个计数器的值。 下面是一个示例代码,可用于实现1秒钟的自动增减: ```c #include "stm32f4xx.h" uint32_t counter = 0; uint8_t increment = 1; void TIM3_IRQHandler(void) { if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) { TIM_ClearITPendingBit(TIM3, TIM_IT_Update); if (increment) { counter++; } else { counter--; } if (counter == 10) { increment = 0; } else if (counter == 0) { increment = 1; } } } int main(void) { RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_TimeBaseStructure.TIM_Prescaler = 41999; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseStructure.TIM_Period = 1999; TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_RepetitionCounter = 0; TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); NVIC_InitTypeDef NVIC_InitStructure; NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE); TIM_Cmd(TIM3, ENABLE); while (1) { // Do nothing } } ``` 在上述代码中,我们首先定义了一个计数器变量counter以及一个布尔变量increment。counter用于记录当前的计数器值,increment则用于指示计数器是应该增加还是减少。在每次中断事件发生时,我们检查increment的值,并根据其值增加或减少counter的值。如果counter达到了10,则我们将increment的值设置为0,表示计数器应该开始减少。如果counter达到了0,则我们将increment的值设置为1,表示计数器应该开始增加。 在main函数中,我们首先配置了定时器TIM3,并使用中断处理函数TIM3_IRQHandler来处理定时器中断事件。然后,我们启用了定时器中断,并启动了定时器。 在while循环中,我们什么也不做,因为我们的定时器将在后台自动工作。当计数器的值发生变化时,我们可以通过读取counter的值来获取最新的计数器值。 总的来说,本文介绍了如何使用STM32定时器实现1秒钟的自动增减。虽然本例中只是简单地使用一个计数器变量来记录时间,但是您可以使用类似的方法来实现更复杂的时间测量、控制等功能。
评论 30
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值