Loading

简述 K60 直流电机闭环控制

仅用于备份本人所写笔记,如有错误或不完善之处还请包含。转载请注明出处!

程序使用逐飞科技 K60 库 V2.0.3

简述方法

电机速度闭环控制,简单来说就是将电机的输出量(速度)通过编码器反馈到控制端(单片机),然后对该输出量和设定的输入量进行比对,接着经过 PID 运算,将运算结果输入给电机的过程。

电机速度闭环一个显而易见的好处就是,可以让小车在不同的路段动态调整小车的速度,如果参数调整的好,可以让小车极大的提升对赛道的适应性能,减少小车跑完一圈所需的时间。

闭环控制思路比较简单,首先需要获取编码器的值,每 4ms 取一次值(可以设定其他时间),然后将取到的值进行相应的计算,得出电机当前速度。

代码示例:

const float unit_pulse = 1783.0;  // 小车轮胎旋转一周(360°)的脉冲
const float motor_radius = 0.063;  // 电机轮胎半径(米)

// 编码器初始化
void EncoderInit(void) {
    ftm_quad_init(ftm2);  // 初始化 ftm2 为正交解码

    port_init_NoAlt(B18, PULLUP);  // 上拉
    port_init_NoAlt(B19, PULLUP);  // 上拉
}

// 读取编码器值
float EncoderRead(void) {
    float data = 0;

    data = ftm_quad_get(ftm2);  // 获取编码器的脉冲值
    ftm_quad_clean(ftm2);       // 清除正交解码的脉冲值

    return data;
}

// 计算电机当前速度
// 输入值为运行该程序的间隔时间
float GetMotorSpeed(uint16 run_time_ms) {
    float motor_speed = 0;

    encoder.finalValue +=
        0.25f * (encoder.value - encoder.finalValue);  // 一阶低通滤波
    encoder.motorRounds =
        (encoder.finalValue * run_time_ms) / unit_pulse;  // 电机每 1s 转的圈数
    encoder.motorDistance =
        encoder.motorRounds * (2 * PI * motor_radius);  // 电机运行长度 m
    motor_speed = encoder.motorDistance / 1;            // 电机速度 m/s

    return motor_speed;
}

计算出速度后,把该变量赋值给 PID 计算,算出需要输出给电机的占空比。PID 计算公式可以根据实际情况稍作修改优化。

代码示例:

// 电机 pid 程序
void MotorPid(void) {
    static float err, last_err, expect_pwm;

    // 请根据实际情况修改 KP、KI 参数大小
    motor_pid.kp = 0.1;
    motor_pid.ki = 0.2;

    err = motor.expectSpeed - encoder.motorSpeed;
    expect_pwm += motor_pid.kp * (err - last_err) + motor_pid.ki * err;
    last_err = err;

    // 电机占空比限幅
    // 请根据实际情况修改最大和最小 PWM 参数值
    if (expect_pwm > 260) expect_pwm = 260;
    if (expect_pwm <= 0) expect_pwm = 0;

    // 将值转换为正整数类型
    motor.expectDutyRatio = (uint16)(expect_pwm);
}

PID 返回期望占空比赋值给电机控制函数:

// 电机控制程序
void MotorControl(void) {
    MotorPid();
    MotorPWM(0, motor.expectDutyRatio);
}

这样就实现了简单的电机速度闭环控制。当然这只算是比较基础的闭环,若要实现复杂功能,还需进行大量优化。

代码

如下为本文闭环全部代码。

motor.c

/*******************************************************************************************
 * COPYRIGHT NOTICE
 * Copyright (c) 2019, ZhangXiao
 * All rights reserved.
 *
 * 本文件仅供参考交流,未经允许不得用于商业用途!
 *
 * File           : motor.c
 * Author         : ZhangXiao
 * Blog           : www.cnblogs.com/zhangxiaochn
 * Version        : v1.0.1
 * Date           : 2019-08-06
 * Software       : IAR 7.70.1
 * Description    : 电机相关程序以及 PID、编码器程序等
 * note           : None
********************************************************************************************/

#include "motor.h"
#include "headfile.h"

const float unit_pulse = 1783.0;  // 小车轮胎旋转一周(360°)的脉冲
const float motor_radius = 0.063;  // 电机轮胎半径(米)

// 电机 pid 程序结构体参数
struct MOTOR_PID motor_pid = {0.1, 0.2, 0, 0, 0, 0, 0, 0};

// 编码器结构体
struct ENCODER encoder = {0, 0, 0, 0, 0};

// 电机结构体
struct MOTOR motor = {0, 0, 0};

// 电机初始化
// FTM1_CH0_PIN 端口为 A12
// FTM1_CH1_PIN 端口为 A13
// 电机初始化频率默认为 10k,请根据实际情况更改频率大小
void MotorInit(void) {
    ftm_pwm_init(ftm1, ftm_ch0, 10 * 1000, 0);
    ftm_pwm_init(ftm1, ftm_ch1, 10 * 1000, 0);
}

// 电机 PWM 控制
void MotorPWM(uint16 pwm_1, uint16 pwm_2) {
    ftm_pwm_duty(ftm1, ftm_ch0, pwm_1);  // A12 端口
    ftm_pwm_duty(ftm1, ftm_ch1, pwm_2);  // A13 端口
}

// 编码器初始化
void EncoderInit(void) {
    ftm_quad_init(ftm2);  // 初始化 ftm2 为正交解码

    port_init_NoAlt(B18, PULLUP);  // 上拉
    port_init_NoAlt(B19, PULLUP);  // 上拉
}

// 读取编码器值
float EncoderRead(void) {
    float data = 0;

    data = ftm_quad_get(ftm2);  // 获取编码器的脉冲值
    ftm_quad_clean(ftm2);       // 清除正交解码的脉冲值

    return data;
}

// 计算电机当前速度
// 输入值为运行该程序的间隔时间
float GetMotorSpeed(uint16 run_time_ms) {
    float motor_speed = 0;

    encoder.finalValue +=
        0.25f * (encoder.value - encoder.finalValue);  // 一阶低通滤波
    encoder.motorRounds =
        (encoder.finalValue * run_time_ms) / unit_pulse;  // 电机每 1s 转的圈数
    encoder.motorDistance =
        encoder.motorRounds * (2 * PI * motor_radius);  // 电机运行长度 m
    motor_speed = encoder.motorDistance / 1;            // 电机速度 m/s

    return motor_speed;
}

// 计算电机运行距离
// 输入值为运行该程序的间隔时间
float GetMotorDistance(uint16 run_time_ms) {
    return encoder.motorDistance / run_time_ms;
}

// 编码器相关参数计算
// 输入值为运行该程序的间隔时间
// 程序建议每 4ms 运行一次,即 Encoder_Calculate(4);
void EncoderCalculate(uint16 run_time_ms) {
    uint16 run_time = 1000 / run_time_ms;  // 每秒钟运行此程序次数

    encoder.value = EncoderRead();                 // 读取编码器的值
    encoder.motorSpeed = GetMotorSpeed(run_time);  // 计算电机当前速度
    motor.distance += GetMotorDistance(run_time);  // 获取电机距离
}

// 电机 pid 程序
void MotorPid(void) {
    static float err, last_err, expect_pwm;

    // 请根据实际情况修改 KP、KI 参数大小
    motor_pid.kp = 0.1;
    motor_pid.ki = 0.2;

    err = motor.expectSpeed - encoder.motorSpeed;
    expect_pwm += motor_pid.kp * (err - last_err) + motor_pid.ki * err;
    last_err = err;

    // 电机占空比限幅
    // 请根据实际情况修改最大和最小 PWM 参数值
    if (expect_pwm > 260) expect_pwm = 260;
    if (expect_pwm <= 0) expect_pwm = 0;

    // 将值转换为正整数类型
    motor.expectDutyRatio = (uint16)(expect_pwm);
}

// 电机控制程序
void MotorControl(void) {
    MotorPid();
    MotorPWM(0, motor.expectDutyRatio);
}

motor.h

#ifndef _MOTOR_H
#define _MOTOR_H

extern struct MOTOR_PID motor_pid;   // PID 结构体
extern struct ENCODER encoder;       // 编码器结构体
extern struct MOTOR motor;           // 电机结构体

void MotorInit(void);                       // 电机初始化
void MotorControl(void);                    // 电机控制

void  EncoderInit(void);                    // 编码器初始化
float EncoderRead(void);                    // 编码器取脉冲值

void  GetMotorSpeed(void);                  // 获取电机速度
float GetMotorDistance(void);               // 获取电机距离

#endif

math.h

// PID 变量
struct MOTOR_PID {

    float kp;
    float ki;
    float kd;
    float err;              // 偏差角度
    float lastErr;          // 上一个偏差
    float beforeErr;        // 上上个偏差
    float addErr;           // 误差累加
    float finalVal;         // 最终得出的值
};

// 电机
struct MOTOR {

    float expectSpeed;       // 电机期望速度
    uint16 expectDutyRatio;  // 电机期望占空比
    float distance;          // 电机运行距离

};

// 编码器
struct ENCODER {

    float value;           // 编码器的值
    float finalValue;      // 编码器最终值
    float motorRounds;     // 电机 5ms 转的圈数
    float motorDistance;   // 电机运行距离
    float motorSpeed;      // 电机速度

};
posted @ 2019-08-06 14:08  滑稽果  阅读(821)  评论(0编辑  收藏  举报
浏览器标题切换
浏览器标题切换end