旋转编码器控制舵机同步运动

1、旋转编码器介绍

旋转编码器:用来测量位置、速度或旋转方向的装置,当其旋转轴旋转时,其输出端可以输出与旋转速度和方向对应的方波信号,读取方波信号的频率和相位信息即可得知旋转轴的速度和方向

2、舵机介绍

本项目使用最常见的SG90舵机

舵机是一种根据输入PWM信号占空比来控制输出角度的装置

输入PWM信号要求:周期为20ms,高电平宽度为0.5ms~2.5ms

该舵机可以实现0~180°转动 ,且转动角度与高电平宽度成线性关系

故我们若把上述图片中-90°角看为0°角,可以得到高电平宽度与角度之间的关系是

                                        Angle = (高电平宽度 - 0.5)* 90

3、STM32F103C8T6内部资源分配及接线

旋转编码器:A相接PA6,B相接PA7,使用定时器3的编码器接口(通道1和通道2)

舵机:接PA0,使用定时器2的通道1

OLED屏:SCK—PB8,SDA—PB9

4、代码展示(完整代码可在主页资源处下载)

Encoder.c

#include "stm32f10x.h"                  // Device header


void Encoder_Init(void)
{
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	
	
	GPIO_InitTypeDef GPIO_InitStructure;
 	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;  
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
 	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
	TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
	TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
	TIM_TimeBaseInitStructure.TIM_Period = 65536-1;       //ARR
	TIM_TimeBaseInitStructure.TIM_Prescaler = 1-1;     //PSC
	TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;
	TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);

	TIM_ICInitTypeDef TIM_ICInitStructure;
	TIM_ICStructInit(&TIM_ICInitStructure);
	TIM_ICInitStructure.TIM_Channel = TIM_Channel_1;
	TIM_ICInitStructure.TIM_ICFilter = 0xF;
	TIM_ICInit(TIM3,&TIM_ICInitStructure);
	TIM_ICInitStructure.TIM_Channel = TIM_Channel_2;
	TIM_ICInitStructure.TIM_ICFilter = 0xF;
	TIM_ICInit(TIM3,&TIM_ICInitStructure);

	TIM_EncoderInterfaceConfig(TIM3,TIM_EncoderMode_TI12,TIM_ICPolarity_Rising,TIM_ICPolarity_Rising);

	TIM_Cmd(TIM3,ENABLE);
}

int16_t Encoder_Get(void)
{
	return TIM_GetCounter(TIM3);
}

PWM.c

#include "stm32f10x.h"                  // Device header


void PWM_Init(void)
{
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	
	TIM_InternalClockConfig(TIM2);
	
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
	TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
	TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
	TIM_TimeBaseInitStructure.TIM_Period = 20000-1;		//ARR
	TIM_TimeBaseInitStructure.TIM_Prescaler = 72-1;		//PSC
	TIM_TimeBaseInitStructure.TIM_RepetitionCounter = 0;
	TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStructure);
	
	TIM_OCInitTypeDef TIM_OCInitStructure;
	TIM_OCStructInit(&TIM_OCInitStructure);
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
	TIM_OCInitStructure.TIM_Pulse = 0;		//CCR
	TIM_OC2Init(TIM2,&TIM_OCInitStructure);
	
	TIM_Cmd(TIM2,ENABLE);
}

void PWM_SetCompare2(uint16_t Compare)
{
	TIM_SetCompare2(TIM2,Compare);
}

Servo.c

#include "stm32f10x.h"                  // Device header
#include "PWM.h"

void Servo_Init(void)
{
	PWM_Init();
}

void Servo_SetAngle(uint16_t Angle)
{
	PWM_SetCompare2(2000 - 100 / 9 * Angle);
}

### 使用旋转编码器调整PWM频率的方法及实现方案 #### 1. 硬件准备 为了实现这一目标,需要具备如下硬件组件: - EC11 编码器模块用于提供旋转输入信号[^2]。 - 支持 PWM 输出的微控制器或开发板,例如 STC 单片机或其他兼容设备。 #### 2. 软件设计思路 软件部分主要分为两个方面的工作: ##### 数据采集与处理 通过读取来自旋转编码器的方向和增量脉冲,计算当前转动的角度变化量。对于 LabVIEW 用户来说,可以通过配置 NI9401 这样的高速数字 I/O 模块来捕捉这些脉冲信号并解析成有用的信息[^1]。 ##### PWM 控制逻辑 根据所获得的位置信息动态改变 PWM 的周期参数从而达到调整输出频率的效果。需要注意的是,在实际操作过程中应当确保系统的响应速度能够跟上编码器的变化速率以免造成失步现象。 ```c // 假设使用 C 语言编写 STM32 或其他 MCU 上的应用程序 void adjust_pwm_frequency(int new_position){ static int last_pos = 0; if(new_position != last_pos){ // 只有当位置发生变化时才更新PWM设置 float delta = (float)(new_position - last_pos); // 计算新的PWM频率值,这里只是一个简单的线性映射例子 uint32_t pwm_freq = base_freq + scale_factor * delta; set_pwm_frequency(pwm_channel, pwm_freq); last_pos = new_position; } } ``` 此函数 `adjust_pwm_frequency` 收由编码器产生的最新绝对位置作为输入,并据此修改指定通道下的 PWM 形发生器工作于不同的载频之上。其中涉及到了几个关键变量: - `last_pos`: 存储前一次记录到的位置以便比较是否有变动; - `delta`: 表达当前位置对于上次采样时刻之间的差异程度; - `base_freq`, `scale_factor`: 定义了基础频率以及每单位角度增量所带来的频率偏移幅度; 最后调用了假设存在的 API 函数 `set_pwm_frequency()` 来完成具体设定动作。 #### 3. 实验验证 构建上述电路连好各部件之后就可以运行测试程序观察效果了。建议先从小范围内的简单场景做起比如仅允许正向/反向切换而后再逐步增加复杂度直至满足项目需求为止。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值