stm32 霍尔编码器电机测速原理及移动距离计算

本次选用的编码器电机为13线的霍尔编码器电机,电机减速比为30:1,转动一圈输出1330=390个脉冲。轮胎直径为75mm,轮胎周长为pid=375=225mm.定时器采用四倍频计数,则一圈输出3904=1560个脉冲。具体编码器知识这里就不多说了。

根据测速原理:

假设编码器输出的脉冲数为N,而电机转动一圈输出1569个脉冲,转动一圈轮子将前进225mm。那输出脉冲数为N时前进的距离就应该为225*(N/1560)mm,再除以时间及可得速度。

下面为具体代码:

encoder.c文件

#include "encoder.h"
 
void Encoder_TIM2_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);		//开时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStruct;							//配置IO口
	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;
	GPIO_Init(GPIOA, &GPIO_InitStruct);
	
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;				//定时器初始化
	TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;
	TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;
	TIM_TimeBaseInitStruct.TIM_Period = 65535;
	TIM_TimeBaseInitStruct.TIM_Prescaler = 0;
	TIM_TimeBaseInit(TIM2, &TIM_TimeBaseInitStruct);
	
	//定时器编码器模式初始化
	TIM_EncoderInterfaceConfig(TIM2, TIM_EncoderMode_TI12, TIM_ICPolarity_Rising, TIM_ICPolarity_Rising);
	
	TIM_ICInitTypeDef TIM_ICInitStruct; 						//输入捕获单元配置
	TIM_ICStructInit(&TIM_ICInitStruct);
	TIM_ICInitStruct.TIM_ICFilter = 10;
	TIM_ICInit(TIM2, &TIM_ICInitStruct);
	
	TIM_ClearFlag(TIM2, TIM_FLAG_Update);
	
	TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);
	
	TIM_SetCounter(TIM2, 0);
	
	TIM_Cmd(TIM2, ENABLE);
}
 
void Encoder_TIM4_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStruct;
	GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;
	GPIO_InitStruct.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
	GPIO_Init(GPIOB, &GPIO_InitStruct);
	
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
	TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;
	TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;
	TIM_TimeBaseInitStruct.TIM_Period = 65535;
	TIM_TimeBaseInitStruct.TIM_Prescaler = 0;
	TIM_TimeBaseInit(TIM4, &TIM_TimeBaseInitStruct);
	
	TIM_EncoderInterfaceConfig(TIM4, TIM_EncoderMode_TI12, TIM_ICPolarity_Rising, TIM_ICPolarity_Rising);
	
	
	TIM_ICInitTypeDef TIM_ICInitStruct; 
	TIM_ICStructInit(&TIM_ICInitStruct);
	TIM_ICInitStruct.TIM_ICFilter = 10;
	TIM_ICInit(TIM4, &TIM_ICInitStruct);
	
	TIM_ClearFlag(TIM4, TIM_FLAG_Update);
	
	TIM_ITConfig(TIM4, TIM_IT_Update, ENABLE);
	
	TIM_SetCounter(TIM4, 0);
	
	TIM_Cmd(TIM4, ENABLE);
}
 
int Read_Spead(int TIMx)			//读取编码器输出脉冲数
{
	int value_1;
	switch(TIMx)
	{
		case 2:value_1 = (short)TIM_GetCounter(TIM2);TIM_SetCounter(TIM2, 0);break;
		case 4:value_1 = (short)TIM_GetCounter(TIM4);TIM_SetCounter(TIM4, 0);break;
		default:value_1 = 0;
	}
	return value_1;
}
 
void TIM2_IRQHander(void)
{
	if(TIM_GetITStatus(TIM2, TIM_IT_Update) == 1)
	{
		TIM_ClearITPendingBit(TIM2, TIM_IT_Update);
	}
}
 
void TIM4_IRQHander(void)
{
	if(TIM_GetITStatus(TIM4, TIM_IT_Update) == 1)
	{
		TIM_ClearITPendingBit(TIM4, TIM_IT_Update);
	}
}

我使用的是简单的delay一下来采集定时器捕获的编码器脉冲数,不过我建议使用定时器中断来处理编码器采集。此函数只采集右轮脉冲进行计算

主函数循环体内函数:

while(1)
{
	delay_s(1);
	{
		uint16_t right = Read_Spead(2);//采集右轮脉冲数
		displacement = 0.225 * (right / 1560);//计算位移
		speed = displacement;//由于我设置的为延时一秒就不用除时间
		OLED_Float(0, 0, speed, 4);//通过OLED显示速度
		set_PWM(999);//设置电机PWM
		TIM_SetCounter(TIM2, 0);//下一次一秒计数前再一次将计数清零
	}
}
### 使用无刷电机编码器计算移动距离 对于无刷电机而言,其工作原理基于转子与定子之间的磁力相互作用使转子旋转并带动负载运动[^1]。为了精确测量由这种旋转引起的线性位移或角度变化,通常会在系统中集成编码器。 #### 编码器的工作机制 编码器能够提供关于轴位置的信息,分为绝对型和增量型两种主要类别: - **绝对型编码器**:可以直接读取当前位置的角度值。 - **增量型编码器**:每次产生固定数量的脉冲信号对应一定的角位移量;这些脉冲可以被计数以确定相对起始点的位置改变情况。 当涉及到计算实际物理空间中的直线路径长度时,则需考虑齿轮比、轮径等因素的影响。 假设已知条件如下: - 脉冲分辨率 \( P \),即每圈产生的总脉冲数目; - 驱动轮直径 \( D \) (单位: 米); - 测得的实际脉冲数 \( N_p \). 那么可以通过下面的方式得出所走过的路程\( S \): \[ S = \frac{N_p}{P} * πD \] 这里的关键在于理解每个脉冲代表了多少度的转动以及这会转化为多少米的前进距离。因此,如果知道上述参数的具体数值就可以很方便地利用此公式来进行转换了。 另外值得注意的是,在某些应用场景下可能还需要考虑到传动效率损失等问题,但这取决于具体的机械设计细节[^2]。 ```python def calculate_distance(pulses, pulse_per_revolution, wheel_diameter): """ 计算给定脉冲下的行驶距离 参数: pulses : int 实际测得的脉冲总数 pulse_per_revolution : float 每一圈对应的脉冲数 wheel_diameter : float 车轮直径(单位:m) 返回: distance : float 行驶的距离(单位:m) """ pi = 3.141592653589793 circumference = pi * wheel_diameter revolutions = pulses / pulse_per_revolution distance = revolutions * circumference return distance ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值