位置式PID、增量式PID与经典PID控温效果比较

2020-01-12 17:28发布

本帖最后由 coleyao 于 2012-6-7 20:20 编辑

  理想的控温曲线很多人都了解是个啥样子,但要是想用PID控温,得到理想的控温曲线就没那么容易了(通常认为延时越大,升温越快就越不容易控制)。
不过有了PID温控仿真器之后,PID温控就变得很简单了,我在附件中列举了在这三种PID控温方式得到较理想控温曲线时的PID参数(这些参数在升温速率快慢、延时长短、设定温度距离环境温度远近等各种场合适应性较广),大家有空可以自己确认一下(仿真器中温度采集和控温的周期均为1秒)或者亦可尝试寻找一下各种PID控制方式下最佳的PID参数,这三种PID控制方式我个人觉得经典式PID效果最差,尽管在大延时,高升温速率下也能稳定,但控温时间长,波动较大;位置式PID次之,在升温速率30度每分钟,延时达20秒时仍有较理想的控温曲线,缺点是继续增加升温速率或延时时间则会出现振荡;增量式PID最佳,在升温速率40度每分钟,延时达20秒时仍有较理想的控温曲线,缺点同位置式PID(当然这些是我通过仿真软件得到的结论,本人并没有用增量式或位置式PID解决过实际问题,写这个仿真软件的目的一方面是为了交流和学习,另一方面是为了将PID控制的效果和其它方法如PreF_v方法作个比较,我目前在温控系统中实际用过的是改进型的PreF_v方法)。
  当然,我个人的看法是没有最好,只有更好,使用PreF_v控制方法(仅有两个参数PreK和FdK需要调整,并且本身就是时域的方法,而不是像PID那样用频域方法解决时域问题)在升温速率达到70度每分钟,延时达100秒时仍能得到较理想的控温曲线。(注:仿真软件在51分论坛上可以下载,效果请各网友自行评估。)
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
49条回答
FuARM
2020-01-18 14:20
这是本人按照公式写的一个增量式PID的程序,和测试程序,不知正确与否,恳请懂的人指出错误和建议。
想用在小车直流电机的控制上来,直流电机上有挡光片测速。MCU PWM调节转速。

按照推荐的调节PID的步骤,先将Ki=Kp*T/Ti置位0,将Kd=Kp*Td/T=0,当作纯比例控制器来调节,
因此让Ti=10000000.0(足够大就行,让Ki接近0),Td=0。此时A=Kp,B=-Kp;C=0;
增量计算(根据后面的程序改的伪代码):
        e_0 = Set - Get;//目标值-本次测量值
        delta = A*e_0 + B*e_1 + C*e_2 = Kp(e_0 - e_1) ;
        e_2 = e_1;
        e_1 = e_0;
        out += delta;
那么,按照以上推导,在纯比例控制中,增量=Kp*(前两次误差的差值);
我的疑问是:
        假设目标值是100,初始状态e_0=0,e_1=0;
        时刻1测量值是10,那么误差就是90,e_0=90,e_1=0;那么delta增量会有一个值,同时实际输出量out也有一个值
        如果时刻2测量值还是10,那么e_0=90,e_1=90; Kp(e_0 - e_1) =0;delta增量=0,此时out不增加。
        试想,当输出量out还不足以驱动电机或者电机被负载堵转时,误差恒定不变,但是delta却等于0,输出out不变,电机也不会增加力矩,这似乎有悖我们控制电机转速的初衷。

以上是我的理解,不知道正确么,恳请大家指导。

/************************************************************************/
#include <math.h>

#define Kp   (1.0)                //PID调节的比例常数
#define Ti   (10000000.0)       //PID调节的积分常数
#define Td   (0.0)              //PID调节的微分时间常数
#define T    (0.1)              //采样周期

#define AA  (Kp * (1 + (T / Ti) + (Td / T)))   //A
#define BB   ((-Kp) * (1+(2 * Td / T)))        //B
#define CC   (Kp * Td / T)                    //C

//误差的阀值,小于这个数值的时候,不做PID调整,避免误差较小时频繁调节引起震荡
#define Emin 3

//调整值限幅,防止积分饱和
#define Umax 20
#define Umin -20

//输出值限幅
#define Pmax 245
#define Pmin 0


typedef struct PID
{
        int Set;  //目标值
        int Get;  //当前测量值

        int e_1;
        int e_2;
        
        double A;
        double B;
        double C;
        
        int delta;
        int out;
}PID_t;


void pid_init(PID_t * pPID);
void pid_ctrl(PID_t * pPID);


void pid_init(PID_t * pPID)
{
        pPID->Set = 0;
        pPID->Get = 0;
        pPID->e_1 = 0;
        pPID->e_2 = 0;
        pPID->delta = 0;
        pPID->out = 0;
        
        pPID->A=AA;
        pPID->B=BB;
        pPID->C=CC;
}


void pid_ctrl(PID_t * pPID)
{

        int e_0 = pPID->Set - pPID->Get;//求差值
        
        pPID->delta = (int)(pPID->A*e_0 + pPID->B*pPID->e_1 + pPID->C*pPID->e_2);
        pPID->e_2 = pPID->e_1;
        pPID->e_1 = e_0;
        pPID->out += pPID->delta;
}


PID_t PID_A;

void main(void)
{
        int i;
        pid_init(&PID_A);
        
        printf("Kp=%f,Ki=%f,Kd=%f ",Kp,(Kp*T/Ti),(Kp*Td/T));
        printf("A=%f,B=%f,C=%f ",AA,BB,CC);

        printf("input the set val ");
        scanf("%d",&PID_A.Set);
        printf("Set=%d ",PID_A.Set);


        while(1){
                scanf("%d",&PID_A.Get);
                pid_ctrl(&PID_A);
                printf("get=%8d ",PID_A.Get);
                printf("delta=%8d, out=%8d ",PID_A.delta,PID_A.out);
                printf("e_1  =%8d, e_2=%8d ",PID_A.e_1,PID_A.e_2);
        
        }        
        printf("leave");
}




一周热门 更多>