【STM32】【HAL库】定时器编码器模式测速

目录

概述

HAL设置

 定时器的编码器模式

定时器设置

 常用函数

代码


概述

电机AB相增量型编码器的介绍和解码方法在这里介绍过了

电机编码器https://blog.csdn.net/m0_57585228/article/details/125791283

测速可以使用外部中断进行脉冲计数

很多型号的单片机中有专门的电路来计算脉冲的速度和方向,也就是定时器的编码器模式

因为是硬件计数,所以计数频率可以很高,性能比软件来的好,使用也比较方便

但是需要连接到指定的GPIO

这里以常见的4倍频测速 为例子进行设置

HAL设置

要开启:

定时器的编码器模式

定时器中断

一个周期性的定时器及中断

 定时器的编码器模式

开启编码器模式

 编码器的设置

  

 解释一下

Polarity:是表示在哪个边沿(上升沿或者下降沿)更新数值

这个比较容易被误解,不管选择哪个,都是会在两种边沿进行计数

只是表示在哪个边沿进行更新数值

定时器设置

间隔一个固定的时间产生中断,这里设置的是10ms

 常用函数

看这个的对应部分https://mp.csdn.net/mp_blog/creation/editor/125279297

代码

在main的初始化后循环之前加入

    HAL_TIM_Encoder_Start(&htim2, TIM_CHANNEL_ALL); //开启编码器模式
    HAL_TIM_Base_Start_IT(&htim2);                  //开启编码器的中断
    HAL_TIM_Base_Start_IT(&htim3);                  //开启定时器的中断

中断回调函数

int32_t Speed = 0;
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
    int16_t zj;
    if (htim == &htim2) //编码器
    {
    }
    else if (htim == &htim3) //计算转速10ms
    {
        zj = __HAL_TIM_GetCounter(&htim2);//获取计数值
        __HAL_TIM_SetCounter(&htim2, 0);//清空计数值
        Speed = (float)zj / (4 * 96) * 100 * 60;//计算转速
        printf("%d\r\n", Speed);//输出转速
    }
}

 在这个函数中根据之前的公式计算处转速

这里要注意:

1.数据范围,不能溢出

2.正数代表正转,负数代表反转

3.需要加入强转(float),因为涉及到了除法,需要避免出现因为舍去位数,导致一直出现0

(或者乘法放到前步)

到最后给速度赋值时在进行丢失小数位(当然也可以直接用小数进行运算)

### 使用STM32 HAL通过定时器实现编码器测速 #### 1. 定时器工作原理 在STM32微控制器中,TIM(Timer)模块可以通过配置为编码器模式来测量旋转速度。时钟源信号经过预分频器分频后传递给计数器,当计数器达到设定的最大值或最小值时会产生溢出或下溢事件[^1]。 #### 2. 编码器模式简介 编码器接口能够根据编码器的旋转方向控制CNT寄存器的计数方向。具体来说,如果编码器正转,则CNT寄存器会自增;如果是反转,则CNT寄存器会自减[^4]。 #### 3. 主要函数说明 以下是几个常用的HAL函数及其功能描述: - **`HAL_TIM_Encoder_GetValue()`**: 获取当前编码器的计数值,可用于计算实时位置信息。 - **`HAL_TIM_Encoder_Start_DMA()`**: 启动DMA传输以捕获编码器信号,减轻CPU负载。 - **`HAL_TIM_IC_CaptureCallback()`**: 当输入捕获事件触发时调用此回调函数,适用于中断方式下的数据处理[^2]。 #### 4. 强制转换的应用 为了简化对电机转向的判断逻辑,在某些情况下可以利用C语言中的强制类型转换机制将无符号整型变量转化为有符号短整形(short),这样可以直接得到带符号的速度值而无需额外运算[^3]。 #### 示例代码 下面提供了一段基于上述理论编写的一个简单例子程序片段: ```c #include "stm32f4xx_hal.h" // 初始化定时器作为编码器模式 void MX_TIM2_Init(void){ TIM_HandleTypeDef htim2; __HAL_RCC_TIM2_CLK_ENABLE(); htim2.Instance = TIM2; htim2.Init.Prescaler = 0; // 预分频系数设为零表示不分频 htim2.Init.CounterMode = TIM_COUNTERMODE_UP; htim2.Init.Period = 65535;// 自由运行模式周期设置最大可能值 if (HAL_TIM_Encoder_Init(&htim2, &sConfig) != HAL_OK) { Error_Handler(); } } int main(){ uint16_t currentCount,lastCount=0,deltaCount; int speed; while(1){ lastCount=currentCount; currentCount=(uint16_t)(short)HAL_TIM_Encoder_GetValue(&htim2); // 转换成signed short deltaCount=currentCount-lastCount; // 计算速度并考虑溢出情况 if(deltaCount>32768){deltaCount-=65536;} else if(deltaCount<-32768){deltaCount+=65536;} speed=deltaCount*frequency/number_of_pulses_per_revolution; printf("Speed:%d RPM\n",speed); HAL_Delay(1000); // 每秒更新一次显示 } } ``` #### 注意事项 以上仅为基本框架示意,请依据实际硬件连接调整参数以及初始化过程。另外还需要注意外部供电电压匹配等问题以免损坏设备。
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值