【STM32F4系列】【HAL库】电机控制(转速和角度)(PID实战1)
文章目录
- 项目目标
- 硬件搭建
- HAL初始化
- 定时器
- PWM
- 编码器
- 定时器中断
- 串口
- 基础驱动
- 获取速度
- 获取角度
- 电机控制
- PID
- 速度环
- 速度环设计
- 速度环调参
- 调试顺序
- P(比例)
- I(积分)
- 总结
- 位置环
- 位置环设计
- 位置环调参
- P调参
- 成品
项目目标
实现电机最常使用的两个功能,转速控制和位置控制
使用PID闭环控制(控制线性系统最简单快捷的控制方法)
硬件搭建
为了实现控制电机转动和闭环控制
需要:
- 电机(废话)
- 编码器(霍尔编码器或者光电编码器均可)
- 电机驱动(这里选的是l298n模块)
千万注意黑色的地线,单片机的地要与12V的地(L298n的地)连接
HAL初始化
定时器
PWM
使用硬件PWM输出,定时器1,输出两路PWM分别代表PWM1和PWM2
设置频率为2.4KHz(约417us),最大占空比5000
使用通道1和2,其余均默认设置
定时器1初始化设置(生成的代码),里开启定时器与PWM输出
HAL_TIM_Base_Start_IT(&htim1);
HAL_TIM_Base_Start(&htim1);
HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_1);
HAL_TIM_PWM_Start(&htim1, TIM_CHANNEL_2);
编码器
使用定时器的编码器模式,双边沿计数,默认设置就可
定时器2的初始化设置里加入,开启编码器模式和定时器
HAL_TIM_Encoder_Start(&htim2, TIM_CHANNEL_ALL); //开启编码器模式HAL_TIM_Base_Start_IT(&htim2);HAL_TIM_Base_Start(&htim2);
定时器中断
每10ms触发一次中断,用于计算PID
注意要打开中断
开启定时器中断
HAL_TIM_Base_Start_IT(&htim3);HAL_TIM_Base_Start(&htim3);
串口
用于调试,默认设置就可,使用printf重定向,无需开启中断
基础驱动
获取速度
定时10ms读取一次编码器的计数值并清零,计算速度
电机是15线霍尔传感器,34:1减速比
详情看这个博客,传送门
float Get_Speed()
{int16_t zj;float Speed = 0;zj = __HAL_TIM_GetCounter(&Encoder_TIM_Handle);__HAL_TIM_SetCounter(&Encoder_TIM_Handle, 0);Speed = (float)zj / (4 * 15 * 34) * 100 * 60;return Speed;
}
获取角度
间隔一段时间读取编码器的计数值(清零操作交由速度获取函数处理)
调用时需要将函数的输出值进行累加
float Get_Angle()
{int16_t zj;float angle = 0;zj = __HAL_TIM_GetCounter(&Encoder_TIM_Handle);angle = (float)zj / (4 * 15 * 34) * 360;return angle;
}
电机控制
通过更改PWM的占空比来控制电机转速
void motor(int16_t Speed)
{if (Speed == 0){__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, Motor_TIM_Channel1, Motor_MAX_Duty + 1);__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, Motor_TIM_Channel2, Motor_MAX_Duty + 1);}else if (Speed > 0){__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, Motor_TIM_Channel1, Speed);__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, Motor_TIM_Channel2, 0);}else if (Speed < 0){Speed *= -1;__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, TIM_CHANNEL_1, 0);__HAL_TIM_SET_COMPARE(&Motor_TIM_Handle, TIM_CHANNEL_2, Speed);}
}
PID
使用增量式PID
PID原理请看,传送门
typedef struct __PID_Increment_Struct
{float Kp, Ki, Kd; //系数float Error_Last1; //上次误差float Error_Last2; //上次误差float Out_Last; //上次输出
} PID_Increment_Struct;float PID_Increment(PID_Increment_Struct *PID, float Current, float Target)
{float err, //误差out, //输出proportion, //比例differential; //微分err = (float)Target - (float)Current; //计算误差proportion = (float)err - (float)PID->Error_Last1; //计算比例项differential = (float)err - 2 * (float)PID->Error_Last1 + (float)PID->Error_Last2; //计算微分项out = (float)PID->Out_Last + (float)PID->Kp * proportion + (float)PID->Ki * err + (float)PID->Kd * differential; //计算PIDPID->Error_Last2 = PID->Error_Last1; //更新上上次误差PID->Error_Last1 = err; //更新误差PID->Out_Last = out; //更新上此输出return out;
}
速度环
速度环设计
速度环就是让电机保持固定转速的PID控制系统
逻辑框图如下,
通过编码器获得转速送到输入作为反馈
输出通过控制PWM(正负和占空比)来控制电机转速
输入的是目标的转速
注意:PID的系数与间隔时间有关,PID需要间隔固定的时间进行调用
那编程的思路就很明显了,我们使用一个定时器中断,在固定的时间(10ms)调用计算一次PID
在这个定时器中断里,我们首先读取转速,之后压入PID进行计算,再将PWM给到电机就行
为了便于观察,这里加上了使用Printf通过串口发送给上位机显示的功能
这里的PID的参数是我调好的
PID_Increment_Struct PID_Speed = {3, 0.6, 0.6};
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{float Speed = 0;int16_t set_speed = 0;if (htim == &htim2){}else if (htim == &htim3){//10ms中断Speed = Get_Speed();//获取转速mb_speed = 3000;set_speed = PID_Increment(&PID_Speed, Speed, mb_speed);//PIDif (set_speed > 5000)set_speed = 5000;else if (set_speed < -5000)set_speed = -5000;//限幅if (set_speed > 500 || set_speed < -500)//死区控制,改善电机异响motor(set_speed);printf("%f,%f\r\n", Speed,mb_speed);//打印当前和目标转速}
}
速度环调参
调试顺序
这个是我用的电机,从某个车模上拆的,带有15线霍尔传感器,34:1减速比,额定电压12V,额定转速350 r/min
速度环是PID控制器,
我们的调整顺序是P->I->D
下面的图.横轴是时间,红线代表的是当前转速,绿线代表目标转速
P(比例)
比例部分是绝对的主力
如果P的极性错误,则电机会反相开到最大转速
我们从小向大调
Kp=1,Ki=0
我们可以看到,电机不转动,只有异响,说明Kp过小(至少一个数量级)
我们增大Kp,令Kp=10,Ki=0
可以看到,电机已经开始转动,但是距离需要的转速过远(Kp在同等数量级了)
我们继续增大Kp,令Kp=30,Ki=0
这时发现,转速已经达到了目标转速的2/3以上
这时我们继续增大Kp
Kp增大到80
发现并没有继续接近目标值很多了
这时再增加Kp也不会更接近目标值了
我们需要引入Ki了
这里放个Kp过大的现象,Kp=700
这种是电机来不及反应造成的
I(积分)
积分项是用于消除静态偏差(也就是Kp在合理范围内变大也无法继续接近目标值的现象)
我们让Kp=30开始调Ki
如果Ki的极性错误,则会出现如下图,即电机来回震荡运动
Kp=80,Ki=1
发现已经可以达到目标值了,回正速度比较慢,我们继续增大Ki(同一数量级)
Kp=80,Ki=5
这时就已经比较完美了,符合了我的要求了
如果自己的要求更高,可以减少步进值慢慢调一下
总结
到了这里,速度环PID我们已经调完了
转速已经可以稳定了
这是调节位置环的前提
位置环
位置环设计
位置环是建立在速度环之上的
使用串级PID进行控制,内环是速度环,外环是位置环
可以加快收敛速度,提高抗干扰能力
我们的策略是当误差大于一圈(>360°或<-360°)时让电机自己以300r/min旋转,不引入PID控制
当误差在一圈内时,使用PID控制
加入位置环的代码如下
PID_Increment_Struct PID_Speed = {3, 0.6, 0.6};
PID_Increment_Struct PID_Angle = {3.1, 0, 0.06};float angle;//角度
int aa = 0;//目标角度
void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{float Speed = 0;int16_t set_speed = 0;float mb_speed;//目标速度if (htim == &htim2){}else if (htim == &htim3){angle += Get_Angle();Speed = Get_Speed();mb_speed = (int16_t)PID_Increment(&PID_Angle, angle, aa);if (PID_Angle.Error_Last1 > 360)mb_speed = 300;else if (PID_Angle.Error_Last1 < -360)mb_speed = -300;// mb_speed = 300;set_speed = PID_Increment(&PID_Speed, Speed, mb_speed);if (set_speed > 5000)set_speed = 5000;else if (set_speed < -5000)set_speed = -5000;if (set_speed > 500 || set_speed < -500)//改善死区motor(set_speed);//printf("%f\r\n", Speed);Speed = aa;printf("%f,%f\r\n", angle, Speed);//输出当前和目标角度}
}
位置环调参
只使用了P即可达到要求
注意,再位置环调整之前,要将速度环调整完毕
下图的横坐标是时间,红线是当前转动角度,绿线是目标角度
P调参
Kp=1时,设置的目标值是4000
前面的直线部分是误差大于一圈的,以固定转速旋转
最后的误差很小了
Kp=3
目标值每隔2s从-400到400
成品
GitHub
电机位置环,串级pid
电机速度环和位置环PID调参教程
【STM32F4系列】【HAL库】电机控制(转速和角度)(PID实战1)相关推荐
- STM32F4系列HAL库配置定时器实验——输入捕获
STM32F4系列HAL库配置定时器实验--输入捕获 输入捕获简单讲解 输入捕获模式可以用来测量脉冲宽度或者测量频率.我们以测量周期和频率为例,用一个简图来说明输入捕获的原理 假定定时器工作在向上计数 ...
- STM32CubeMX | STM32 F1系列HAL库低功耗STOP和STANDBY模式唤醒(RTC时钟唤醒+外部中断唤醒示例)
STM32CubeMX | STM32 F1系列HAL库低功耗STOP和STANDBY模式唤醒(RTC时钟唤醒+外部中断唤醒示例) 目录 STM32CubeMX | STM32 F1系列HAL库低功耗 ...
- STM32F1系列HAL库配置系统时钟
STM32F1系列HAL库配置系统时钟 其实一开始对于时钟我也是知之甚少,在MSP432中我就一直忽视时钟配置,其实也是在STM32学习时落下的病根,现在趁有空补一下. 时钟简单讲解 对于时钟系统,在 ...
- STM32系列(HAL库)——F103C8T6通过MFRC522、RFID射频卡、门禁卡模块读取卡片ID(二)
本文继上一篇:STM32系列(HAL库)--F103C8T6通过MFRC522.RFID射频卡.门禁卡模块读取卡片ID 本文介绍在运用RC522模块时,运用链表结构存储数据的操作 Let's go! ...
- stm32f103c6t6下的HAL库搭建三种低功耗模式及实战分析(stm32通用)
目录 三种低功耗模式介绍 睡眠模式(sleep mode) 停止模式(stop mode) 待机模式(standby mode) 总结 实战测试 个别电路原理图 功耗分析 ADC功耗大解决方案 ADC ...
- ST FOC电机控制同步电角度测试说明
电机控制同步电角度测试说明 前言 在使用ST FOC电机库时,当使用Hall信号作为位置信号时,需要输入同步电角度数据,这个数据根据当前使用电机的特性进行输入,会在每次Hall信号变化时同步电角度,如 ...
- MPU6050(读取原数据、移植DMP、stm32f4、HAL库、KEIL5)
记录一下自己遇到的问题及解决方法,希望能帮助到一些人. 第一步,读取芯片的原始数据.需要注意两点:1.对HAL库提供的IIC读取写入函数进行再包装.(千万不要觉的这步多此一举,后面移植DMP时用得到) ...
- STM32F4 (hal库)ADC+TIM1+DAC的配置
写在前面:感谢XXX大佬的指导,点击可查看他的博客 实现的功能: 用定时器TIM产生PWM波来控制ADC的采样频率,在ADC中断中将采样值直接通过DAC输出.本文主要展示ADC.TIM.DAC的配置( ...
- 【STM32】HAL库-电源控制(低功耗模式)
电源框图 STM32 的电源系统主要分为备份域电路.内核电路以及 ADC 电路三部分 备份域电路 STM32 的 LSE 振荡器.RTC.备份寄存器及备份 SRAM 这些器件被包含进备份域电路中,这部 ...
最新文章
- 干货 | 使用FFT变换自动去除图像中严重的网纹
- html5页面默认的字符集是什么,HTML 字符集
- 是什么牌子_电暖气片什么牌子好
- 十六、深入Python字符串
- 基础才是重中之重~通过人类的生活来学习Delegate
- Vue 安装 live-server
- 前端开发 选择器的优先级 0229
- 安装这些App的注意了!隐私窃取 捆绑推广 已被下架 现在卸载还来得及!
- sqlserver存储过程加锁后怎么解锁_MySQL 的加锁处理,你都了解的一清二楚了吗?...
- 软件烧录的测试方法,烧录测试座使用及保养
- emacs ido模式
- 给女朋友道歉的java代码_有关于向女朋友道歉的经典句子
- mvp的全称_MVP英文全称是什么
- 岛屿数量问题(C实现)
- 1628:完成基于脚本的安装失败(手把手教学)
- 布局阴影shadow的制作
- Problem E: 薪酬计算
- 河北外国语学院计算机宿舍,2021年河北外国语学院新生宿舍条件和宿舍环境图片...
- 神经网络(Neutral Network)
- 根据灰度直方图调整图象亮度