PID算法(一)PID简介

本文介绍了PID控制算法的基础知识,包括比例(P)、积分(I)和微分(D)三个部分的作用,以及如何根据系统特性调整参数。同时,提供了C语言实现的位置式PID和增量式PID算法,并讨论了两者在误差变化快的系统中适用性的差异。通过代码实例,读者可以更好地理解和应用PID控制算法。
摘要由CSDN通过智能技术生成

PID算法简介及实现代码

PID简介

智能车比赛中,用到了PID算法,写下来当一个总结。PID是很经典且应用很广泛的控制算法,依据误差来减少误差。
PID控制原理图PID

PID分为三部分:

  1. P 比例。P增大,可以加快系统响应速度,但是不能从根本上消除静态误差,而且P过大,容易超调。
  2. I 积分 。用求和代替积分。一段时间内误差的累积。但是对于变化很快的系统,积分时间不宜太长,I不宜给的太大,否则会降低系统的稳定。
  3. D 微分。用增量代替微分。D可以预测偏差,就相当于求了一个加速度,可以做一个预判。但是要注意选择合适的微分时间。

基本的PID算法有两种,位置式PID和增量式PID:

位置式PID:
位置式PID公式
增量式PID:
增量式PID公式

位置式PID和增量式PID的区别:

个人经验之谈,对于误差变化很快的系统,选增量式PID比较合适,因为增量式PID只涉及了前后几次的误差,而位置式PID的积分,是一段时间内的误差,如果误差变化很快的话,位置式的PID的I这一项,反倒会引入更大的误差,不如使用增量式PID。
更详细的,可以 看这篇 位置式PID与增量式PID区别浅析

代码实现:

//pid.h
enum pid_mode
{
	PID_POSITION = 0,
	PID_DELTA
};

typedef struct
{
	char mode;//position or delta

	float kp;//P
	float ki;//I
	float kd;//D

	float iout_max;
	float out_max;

	float target;//goal
	float now;//now

	float out_p;
	float out_i;
	float out_d;
	float out;


	float erro[3];
	float Dbuff[3];
}pid_param_t;

void pid_init(pid_param_t * pid, char mode, const float PID[3], float iout_max, float out_max);
float pid_Calc(pid_param_t * pid, float now, float target);
//pid.c
#include "pid.h"

#define LimitMax(input,max)  \
{							 \
	if (input > max)		 \
	{						 \
		input = max;		 \
	}						 \
	if (input < -max)		 \
	{						 \
		input = -max;		 \
	}						 \
							 \
}

void pid_init(pid_param_t * pid ,char mode,const float PID[3],float iout_max,float out_max)
{
	pid->mode = mode;

	pid->kp = PID[0];
	pid->ki = PID[1];
	pid->kd = PID[2];

	pid->out_max = out_max;
	pid->iout_max = iout_max;

	pid->out_p = pid->out_i = pid->out_d = 0;
	pid->out = pid->last_out = pid->before_out = 0;
	pid->Dbuff[0] = pid->Dbuff[1] = pid->Dbuff[2] = 0;
	pid->erro[0] = pid->erro[1] = pid->erro[2] = 0;

}


float pid_Calc(pid_param_t * pid ,float now,float target)
{
	pid->target = target;
	pid->now = now;

	pid->erro[1] = pid->erro[0];
	pid->erro[2] = pid->erro[1];

	pid->erro[0] = pid->target - pid->now;

	if (pid->mode = PID_POSITION)
	{
		pid->Dbuff[0] = pid->erro[0] - pid->erro[1];

		pid->out_p = pid->kp*pid->erro[0];
		pid->out_i += pid->ki*pid->erro[0];
		pid->out_d = pid->kd*pid->Dbuff[0];

		LimitMax(pid->out_i, pid->iout_max);
	
		pid->out = pid->out_p + pid->out_i +  pid->out_d;
	}
	else if (pid->mode = PID_DELTA)
	{

		pid->Dbuff[1] = pid->Dbuff[0];
		pid->Dbuff[2] = pid->Dbuff[1];

		pid->Dbuff[0] = pid->erro[0] - pid->erro[1];

		pid->out_p = pid->kp*pid->Dbuff[0];
		pid->out_i= pid->ki*pid->erro[0];
		pid->out_d = pid->kd*(pid->Dbuff[0]- pid->kd*pid->Dbuff[1]);

		pid->out = pid->out_p + pid->out_i + pid->out_d;

		LimitMax(pid->out, pid->out_max);

	}
	return pid->out;
}



  • 2
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值