PID采样时间精准控制——基于stm32主控

(一)前言

    我们在用到PID算法时,有时候会使用积分分离的抗饱和算法,或是另一类属于增量式的PID算法。这时,需要提供一个PID采样间隔来控制采样率,大部分无人机类工程用到的采样为10ms,视具体情况而定。
    本文提供一个精确控制采样周期的思路,如果您有更好的思路可以分享。
    本文是利用单片机的定时器去定期中断采样使用PID。
    以stm32为例。

(二)初始化定时器:

void time7_init(u16 per,u16 pre)
{
    TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM7,ENABLE);
    TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1;
    TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up;
    TIM_TimeBaseStructure.TIM_Period=per;
    TIM_TimeBaseStructure.TIM_Prescaler=pre;
    TIM_TimeBaseInit(TIM7,&TIM_TimeBaseStructure);

    TIM_ITConfig(TIM7,TIM_IT_Update,ENABLE);
    TIM_Cmd(TIM7,ENABLE);
}

初始化TIM7。

(三)初始中断

void NVIC_INIT(void)
{
    NVIC_InitTypeDef NVIC_InitStructure;
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);
    NVIC_InitStructure.NVIC_IRQChannel=TIM7_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority=3;
    NVIC_Init(&NVIC_InitStructure);
}

(四)写中断服务函数

void TIM7_IRQHandler(void)
{
    if(TIM_GetITStatus(TIM7,TIM_IT_Update)!=RESET)
    {
        TIM_ClearITPendingBit(TIM7,TIM_IT_Update);
    }
		
		pid_roll.Actualangle=roll;    //记录姿态角
		pid_pitch.Actualangle=pitch;
		pid_yaw.Actualangle=yaw;
		
		roll_pitch_yaw_anglePID(0,0,0);  //计算PID控制量
}

在中断服务函数里,主要是进行一次PID中当前值得获取,再进行一次计算

(五)主函数中的使用

	//TIM7采取数据进行PID算法
	time7_init(71,9999);//每10ms进行一次采样并计算
    NVIC_INIT();

当然可以修改time7_init()的参数来修改采样周期

采用此方式,相比于采用延时函数去控制要精确得多,且不用考虑程序进程

  • 10
    点赞
  • 70
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
您好!以下是一个基于STM32PID控制直流电机的代码示例: 首先,需要包含相关的头文件和定义一些变量: ```c #include "stm32f4xx.h" #define MOTOR_PWM_PIN GPIO_Pin_0 #define MOTOR_GPIO_PORT GPIOA #define KP 1.0 #define KI 0.5 #define KD 0.2 float setpoint = 0.0; // 目标设定值 float feedback = 0.0; // 反馈值 float error = 0.0; // 误差值 float prevError = 0.0;// 上一次的误差值 float integral = 0.0; // 积分项 float derivative = 0.0; // 微分项 float output = 0.0; // 控制输出值 ``` 接下来,需要初始化相关的GPIO和定时器: ```c void GPIO_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); GPIO_InitStructure.GPIO_Pin = MOTOR_PWM_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL ; GPIO_Init(MOTOR_GPIO_PORT, &GPIO_InitStructure); GPIO_PinAFConfig(MOTOR_GPIO_PORT, GPIO_PinSource0, GPIO_AF_TIM2); } void TIM_Configuration(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_OCInitTypeDef TIM_OCInitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); TIM_TimeBaseStructure.TIM_Period = 999; // 根据实际情况调整 TIM_TimeBaseStructure.TIM_Prescaler = 83; // 根据实际情况调整 TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OC1Init(TIM2, &TIM_OCInitStructure); TIM_OC1PreloadConfig(TIM2, TIM_OCPreload_Enable); TIM_Cmd(TIM2, ENABLE); } ``` 接下来,实现PID控制算法的函数: ```c void PID_Controller(void) { error = setpoint - feedback; integral += error; derivative = error - prevError; output = KP * error + KI * integral + KD * derivative; prevError = error; } ``` 最后,在主函数不断更新反馈值,并根据PID算法调整输出值: ```c int main(void) { SystemInit(); GPIO_Configuration(); TIM_Configuration(); while(1) { // 更新反馈值,这里需要根据实际情况获取电机的转速或位置 feedback = GetMotorFeedback(); // 调用PID控制算法 PID_Controller(); // 更新PWM的占空比 TIM_SetCompare1(TIM2, (uint16_t) output); } } ``` 请注意,以上代码仅为示例,具体的实现需要根据您的实际情况进行调整。另外,需要根据您的硬件连接情况来配置相关的GPIO和定时器。 希望对您有所帮助!如有任何问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

TaiBai-let

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值