【STM32F4系列】【HAL库】电机控制(转速和角度)(PID实战1)

这篇具有很好参考价值的文章主要介绍了【STM32F4系列】【HAL库】电机控制(转速和角度)(PID实战1)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

项目目标

实现电机最常使用的两个功能,转速控制位置控制

使用PID闭环控制(控制线性系统最简单快捷的控制方法)

硬件搭建

为了实现控制电机转动闭环控制

需要:

  1. 电机(废话)
  2. 编码器(霍尔编码器或者光电编码器均可)
  3. 电机驱动(这里选的是l298n模块)

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

千万注意黑色的地线,单片机的地要与12V的地(L298n的地)连接

HAL初始化

定时器

PWM

使用硬件PWM输出,定时器1,输出两路PWM分别代表PWM1和PWM2

设置频率为2.4KHz(约417us),最大占空比5000

使用通道1和2,其余均默认设置

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

定时器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);

编码器

使用定时器的编码器模式,双边沿计数,默认设置就可

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

定时器2的初始化设置里加入,开启编码器模式和定时器

  HAL_TIM_Encoder_Start(&htim2, TIM_CHANNEL_ALL); //开启编码器模式
  HAL_TIM_Base_Start_IT(&htim2);
  HAL_TIM_Base_Start(&htim2);

定时器中断

每10ms触发一次中断,用于计算PID

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

注意要打开中断

开启定时器中断

  HAL_TIM_Base_Start_IT(&htim3);
  HAL_TIM_Base_Start(&htim3);

串口

用于调试,默认设置就可,使用printf重定向,无需开启中断

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

基础驱动

获取速度

定时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; //计算PID
    PID->Error_Last2 = PID->Error_Last1;                                                                             //更新上上次误差
    PID->Error_Last1 = err;                                                                                          //更新误差
    PID->Out_Last = out;                                                                                             //更新上此输出
    return out;
}

速度环

速度环设计

速度环就是让电机保持固定转速的PID控制系统

逻辑框图如下,

通过编码器获得转速送到输入作为反馈

输出通过控制PWM(正负和占空比)来控制电机转速

输入的是目标的转速

注意:PID的系数与间隔时间有关,PID需要间隔固定的时间进行调用

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

那编程的思路就很明显了,我们使用一个定时器中断,在固定的时间(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);//PID

        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,%f\r\n", Speed,mb_speed);//打印当前和目标转速
    }
}

速度环调参

调试顺序

这个是我用的电机,从某个车模上拆的,带有15线霍尔传感器,34:1减速比,额定电压12V,额定转速350 r/min

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

速度环是PID控制器,

我们的调整顺序是P->I->D

下面的图.横轴是时间,红线代表的是当前转速,绿线代表目标转速

P(比例)

比例部分是绝对的主力

如果P的极性错误,则电机会反相开到最大转速

我们从小向大调

Kp=1,Ki=0

我们可以看到,电机不转动,只有异响,说明Kp过小(至少一个数量级)

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

我们增大Kp,令Kp=10,Ki=0

可以看到,电机已经开始转动,但是距离需要的转速过远(Kp在同等数量级了)

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

我们继续增大Kp,令Kp=30,Ki=0

这时发现,转速已经达到了目标转速的2/3以上

这时我们继续增大Kp

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

Kp增大到80

发现并没有继续接近目标值很多了

这时再增加Kp也不会更接近目标值了

我们需要引入Ki了

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

这里放个Kp过大的现象,Kp=700

这种是电机来不及反应造成的

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

I(积分)

积分项是用于消除静态偏差(也就是Kp在合理范围内变大也无法继续接近目标值的现象)

我们让Kp=30开始调Ki

如果Ki的极性错误,则会出现如下图,即电机来回震荡运动

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

Kp=80,Ki=1

发现已经可以达到目标值了,回正速度比较慢,我们继续增大Ki(同一数量级)

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

Kp=80,Ki=5

这时就已经比较完美了,符合了我的要求了

如果自己的要求更高,可以减少步进值慢慢调一下

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

总结

到了这里,速度环PID我们已经调完了

转速已经可以稳定了

这是调节位置环的前提

位置环

位置环设计

位置环是建立在速度环之上的

使用串级PID进行控制,内环是速度环,外环是位置环

可以加快收敛速度,提高抗干扰能力

我们的策略是当误差大于一圈(>360°或<-360°)时让电机自己以300r/min旋转,不引入PID控制

当误差在一圈内时,使用PID控制

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

加入位置环的代码如下

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

前面的直线部分是误差大于一圈的,以固定转速旋转

最后的误差很小了

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

Kp=3

目标值每隔2s从-400到400

感应电机转速控制,STM32_HAL库,STM32F4系列,stm32,单片机,arm

成品

GitHub

电机位置环,串级pid

电机速度环和位置环PID调参教程文章来源地址https://www.toymoban.com/news/detail-784004.html

到了这里,关于【STM32F4系列】【HAL库】电机控制(转速和角度)(PID实战1)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • STM32CubeMX 直流电机串级PID位置速度控制、HAL库、cubemx、PID、串级PID、位置控制、速度控制、双环控制

    提示:本文章的串级PID位置速度控制,是在前两篇文章速度控制,位置控制的基础上实现的,这一章节中不需要额外的cubemx的配置,只需要写简单的代码即可,复杂的地方在于串级pid的调试过程。 pid是我们在学习单片机中首先要学会的控制算法,而串级pid又是在单pid的基础上

    2024年02月14日
    浏览(53)
  • 【STM32F4系列】【HAL库】【自制库】WS2812(软件部分)(PWM+DMA)

    硬件介绍(PCB设计方案) 模拟时序发送 WS2812是一种异步串行通信,它每一位数据时间是ns级别的 默认是高电平状态 0码:220-380ns高电平+580-1600ns低电平 1码:580-1600ns高电平+220-380ns低电平 复位码:280us低电平 24Bit数据来代表GRB的亮度值 从高位到低位发送,分别按照G-R-B的顺序发送

    2024年02月15日
    浏览(83)
  • 【STM32F4系列】【HAL库】【模块介绍】MPU6050设置与DMP库使用

    MPU6050是一个3轴陀螺仪(测角加速度)和3轴加速度计(测量线加速度)的测量芯片 内部自带运算单元(DMP),可以输出经姿态融合计算后的 四元数 (一种表示旋转的方法) 而且MPU6050的价格较低(10r以下),常被用于精度不高的场合作为姿态感知的芯片 如经典项目平衡车,某年电赛题目风力摆

    2024年02月05日
    浏览(43)
  • STM32 HAL库PID控制电机 第二章 TB6612FNG芯片驱动GB37-520电机

    1 电路图 2 TB6612简介 TB6612是双驱动,可同时驱动两个电机 STBY:接单片机的IO口清零电机全部停止,置1通过AIN1 AIN2,BIN1,BIN2 来控制正反转 VM:建议接10V以内电源( 瞬间上电12V可能会有尖峰电压击穿器件 ) VCC:接5V电源 GND:接电源负极 PWMA:接单片机的PWM口 ,控制转速 PWM

    2023年04月22日
    浏览(96)
  • 【STM32F429】HAL库的PWM中断,精确控制脉冲数,控制步进电机

    这两天在调步进电机,希望是使得步进电机每次都达到期望的高度。在查了一天的资料,发现大部分上传的资料都是使用CubeMX生成的,可复制性很高,但未免有失可读性,故上传我的心得经验。 本来原子哥的例程里有整合度很高的,已经封装好的精确控制步进电机前进距离的

    2024年02月08日
    浏览(53)
  • STM32(HAL库)——光电编码器、M/T法测量电机转速

    目录 一 编码器 二 电机测试的三种方法 三 STM32CubeMx配置 四 程序编写 五 实验结果 常见的用于电机测速的编码器有霍尔编码器和光电编码器两种。两者测速的基本原理不同,但都是输出两路相位差90°的脉冲信号。这里以光电编码器为例介绍一下测速原理。 光电编码器是由光

    2024年02月07日
    浏览(57)
  • STM32F4驱动42步进电机(采用驱动器)

    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 stm32f407zgt6芯片,tb6600驱动器 植树机代码,可以驱动四路42步进电机,以下是只驱动1路。 : 红线接B+(TB6600), 蓝线接B-, 绿线接A-, 黑线接A+ 该图片是转载的。根据本文代码:Signal的-统一接开发板的GND(解

    2024年02月03日
    浏览(63)
  • 【stm32开发笔记】基于HAL库的STM32F4添加DSP库

    本文分两种方法添加DSP库:1.CubeMX直接配置ioc添加; 2.KEIL内添加; 简述:补齐全部lib库-添加DSP包-使能DSP勾选-添加头文件及魔术棒配置-测试 1.补齐lib库。( 如果使用直接默认添加的库,是不支持FPU的,所以需要补齐后找到所需的lib文件进行替换,在MX的工程管理栏,选择复制所

    2024年02月16日
    浏览(58)
  • FPGA实现电机转速PID控制

             通过纯RTL实现电机转速PID控制,包括电机编码器值读取,电机速度、正反转控制,PID算法,卡尔曼滤波,最终实现对电机速度进行控制,使其能够渐近设定的编码器目标值。         前面通过SOPC之NIOS Ⅱ实现电机转速PID控制(调用中断函数)对电机实现了PID控制,

    2024年02月09日
    浏览(35)
  • MPU6050(读取原数据、移植DMP、stm32f4、HAL库、KEIL5)

    记录一下自己遇到的问题及解决方法,希望能帮助到一些人。 第一步,读取芯片的原始数据。需要注意两点:1、对HAL库提供的IIC读取写入函数进行再包装。(千万不要觉的这步多此一举,后面移植DMP时用得到) 2、芯片的地址(这里面有俩坑)第一就是,芯片的 I2C 设备地址

    2023年04月08日
    浏览(54)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包