首页 > 编程语言 > 详细

Brett Beauregard大神的Arduino PID算法

时间:2019-10-18 10:02:01      阅读:150      评论:0      收藏:0      [点我收藏+]

Improving the Beginner’s PID – Introduction

In conjunction with the release of the new Arduino PID Library I’ve decided to release this series of posts. The last library, while solid, didn’t really come with any code explanation. This time around the plan is to explain in great detail why the code is the way it is. I’m hoping this will be of use to two groups of people:

  • People directly interested in what’s going on inside the Arduino PID library will get a detailed explanation.
  • Anyone writing their own PID algorithm can take a look at how I did things and borrow whatever they like.

It’s going to be a tough slog, but I think I found a not-too-painful way to explain my code.  I’m going to start with what I call “The Beginner’s PID.”  I’ll then improve it step-by-step until we’re left with an efficient, robust pid algorithm.

The Beginner’s PID

Here’s the PID equation as everyone first learns it:

技术分享图片

 

 

This leads pretty much everyone to write the following PID controller:

 1 /*working variables*/
 2 unsigned long lastTime;
 3 double Input, Output, Setpoint;
 4 double errSum, lastErr;
 5 double kp, ki, kd;
 6 void Compute()
 7 {
 8    /*How long since we last calculated*/
 9    unsigned long now = millis();
10    double timeChange = (double)(now - lastTime);
11   
12    /*Compute all the working error variables*/
13    double error = Setpoint - Input;
14    errSum += (error * timeChange);
15    double dErr = (error - lastErr) / timeChange;
16   
17    /*Compute PID Output*/
18    Output = kp * error + ki * errSum + kd * dErr;
19   
20    /*Remember some variables for next time*/
21    lastErr = error;
22    lastTime = now;
23 }
24   
25 void SetTunings(double Kp, double Ki, double Kd)
26 {
27    kp = Kp;
28    ki = Ki;
29    kd = Kd;
30 }

 

 

Compute() is called either regularly or irregularly, and it works pretty well. This series isn’t about “works pretty well” though. If we’re going to turn this code into something on par with industrial PID controllers, we’ll have to address a few things:

  1. Sample Time – The PID algorithm functions best if it is evaluated at a regular interval. If the algorithm is aware of this interval, we can also simplify some of the internal math.
  2. Derivative Kick – Not the biggest deal, but easy to get rid of, so we’re going to do just that.
  3. On-The-Fly Tuning Changes – A good PID algorithm is one where tuning parameters can be changed without jolting the internal workings.
  4. Reset Windup Mitigation –We’ll go into what Reset Windup is, and implement a solution with side benefits
  5. On/Off (Auto/Manual) – In most applications, there is a desire to sometimes turn off the PID controller and adjust the output by hand, without the controller interfering
  6. Initialization – When the controller first turns on, we want a “bumpless transfer.” That is, we don’t want the output to suddenly jerk to some new value
  7. Controller Direction – This last one isn’t a change in the name of robustness per se. it’s designed to ensure that the user enters tuning parameters with the correct sign.
  8. NEW: Proportional on Measurement – Adding this feature makes it easier to control certain types of processe。

Once we’ve addressed all these issues, we’ll have a solid PID algorithm. We’ll also, not coincidentally, have the code that’s being used in the lastest version of the Arduino PID Library. So whether you’re trying to write your own algorithm, or trying to understand what’s going on inside the PID library, I hope this helps you out. Let’s get started.

UPDATE: In all the code examples I’m using doubles. On the Arduino, a double is the same as a float (single precision.) True double precision is WAY overkill for PID. If the language you’re using does true double precision, I’d recommend changing all doubles to floats.

 

Improving the Beginner’s PID – Sample Time

(This is Modification #1 in a larger series on writing a solid PID algorithm)

The Problem

The Beginner’s PID is designed to be called irregularly. This causes 2 issues:

  • You don’t get consistent behavior from the PID, since sometimes it’s called frequently and sometimes it’s not.
  • You need to do extra math computing the derivative and integral, since they’re both dependent on the change in time.

The Solution

Ensure that the PID is called at a regular interval. The way I’ve decided to do this is to specify that the compute function get called every cycle. based on a pre-determined Sample Time, the PID decides if it should compute or return immediately.

Once we know that the PID is being evaluated at a constant interval, the derivative and integral calculations can also be simplified. Bonus!

 1 /*working variables*/
 2 unsigned long lastTime;
 3 double Input, Output, Setpoint;
 4 double errSum, lastErr;
 5 double kp, ki, kd;
 6 int SampleTime = 1000; //1 sec
 7 void Compute()
 8 {
 9    unsigned long now = millis();
10    int timeChange = (now - lastTime);
11    if(timeChange>=SampleTime)
12    {
13       /*Compute all the working error variables*/
14       double error = Setpoint - Input;
15       errSum += error;
16       double dErr = (error - lastErr);
17  
18       /*Compute PID Output*/
19       Output = kp * error + ki * errSum + kd * dErr;
20  
21       /*Remember some variables for next time*/
22       lastErr = error;
23       lastTime = now;
24    }
25 }
26  
27 void SetTunings(double Kp, double Ki, double Kd)
28 {
29   double SampleTimeInSec = ((double)SampleTime)/1000;
30    kp = Kp;
31    ki = Ki * SampleTimeInSec;
32    kd = Kd / SampleTimeInSec;
33 }
34  
35 void SetSampleTime(int NewSampleTime)
36 {
37    if (NewSampleTime > 0)
38    {
39       double ratio  = (double)NewSampleTime
40                       / (double)SampleTime;
41       ki *= ratio;
42       kd /= ratio;
43       SampleTime = (unsigned long)NewSampleTime;
44    }
45 }

On lines 10&11, the algorithm now decides for itself if it’s time to calculate. Also, because we now KNOW that it’s going to be the same time between samples, we don’t need to constantly multiply by time change. We can merely adjust the Ki and Kd appropriately (lines 31 & 32) and result is mathematically equivalent, but more efficient.

one little wrinkle with doing it this way though though. if the user decides to change the sample time during operation, the Ki and Kd will need to be re-tweaked to reflect this new change. that’s what lines 39-42 are all about.

Also Note that I convert the sample time to Seconds on line 29. Strictly speaking this isn’t necessary, but allows the user to enter Ki and Kd in units of 1/sec and s, rather than 1/mS and mS.

The Results

the changes above do 3 things for us

  1. Regardless of how frequently Compute() is called, the PID algorithm will be evaluated at a regular interval [Line 11]
  2. Because of the time subtraction [Line 10] there will be no issues when millis() wraps back to 0. That only happens every 55 days, but we’re going for bulletproof remember?
  3. We don’t need to multiply and divide by the timechange anymore. Since it’s a constant we’re able to move it from the compute code [lines 15+16] and lump it in with the tuning constants [lines 31+32]. Mathematically it works out the same, but it saves a multiplication and a division every time the PID is evaluated

Side note about interrupts

If this PID is going into a microcontroller, a very good argument can be made for using an interrupt. SetSampleTime sets the interrupt frequency, then Compute gets called when it’s time. There would be no need, in that case, for lines 9-12, 23, and 24. If you plan on doing this with your PID implentation, go for it! Keep reading this series though. You’ll hopefully still get some benefit from the modifications that follow.
There are three reasons I didn’t use interrupts

  1. As far as this series is concerned, not everyone will be able to use interrupts.
  2. Things would get tricky if you wanted it implement many PID controllers at the same time.
  3. If I’m honest, it didn’t occur to me. Jimmie Rodgers suggested it while proof-reading the series for me. I may decide to use interrupts in future versions of the PID library.

 

Improving the Beginner’s PID – Derivative Kick

(This is Modification #2 in a larger series on writing a solid PID algorithm)

The Problem

This modification is going to tweak the derivative term a bit. The goal is to eliminate a phenomenon known as “Derivative Kick”.

技术分享图片

The image above illustrates the problem. Since error=Setpoint-Input, any change in Setpoint causes an instantaneous change in error. The derivative of this change is infinity (in practice, since dt isn’t 0 it just winds up being a really big number.) This number gets fed into the pid equation, which results in an undesirable spike in the output. Luckily there is an easy way to get rid of this.

The Solution

技术分享图片
It turns out that the derivative of the Error is equal to negative derivative of Input, EXCEPT when the Setpoint is changing. This winds up being a perfect solution. Instead of adding (Kd * derivative of Error), we subtract (Kd * derivative of Input). This is known as using “Derivative on Measurement”

 1 /*working variables*/
 2 unsigned long lastTime;
 3 double Input, Output, Setpoint;
 4 double errSum, lastInput;
 5 double kp, ki, kd;
 6 int SampleTime = 1000; //1 sec
 7 void Compute()
 8 {
 9    unsigned long now = millis();
10    int timeChange = (now - lastTime);
11    if(timeChange>=SampleTime)
12    {
13       /*Compute all the working error variables*/
14       double error = Setpoint - Input;
15       errSum += error;
16       double dInput = (Input - lastInput);
17  
18       /*Compute PID Output*/
19       Output = kp * error + ki * errSum - kd * dInput;
20  
21       /*Remember some variables for next time*/
22       lastInput = Input;
23       lastTime = now;
24    }
25 }
26  
27 void SetTunings(double Kp, double Ki, double Kd)
28 {
29   double SampleTimeInSec = ((double)SampleTime)/1000;
30    kp = Kp;
31    ki = Ki * SampleTimeInSec;
32    kd = Kd / SampleTimeInSec;
33 }
34  
35 void SetSampleTime(int NewSampleTime)
36 {
37    if (NewSampleTime > 0)
38    {
39       double ratio  = (double)NewSampleTime
40                       / (double)SampleTime;
41       ki *= ratio;
42       kd /= ratio;
43       SampleTime = (unsigned long)NewSampleTime;
44    }
45 }

 

The modifications here are pretty easy. We’re replacing +dError with -dInput. Instead of remembering the lastError, we now remember the lastInput

The Result

技术分享图片

Here’s what those modifications get us. Notice that the input still looks about the same. So we get the same performance, but we don’t send out a huge Output spike every time the Setpoint changes.

This may or may not be a big deal. It all depends on how sensitive your application is to output spikes. The way I see it though, it doesn’t take any more work to do it without kicking so why not do things right?

 

Improving the Beginner’s PID: Tuning Changes

(This is Modification #3 in a larger series on writing a solid PID algorithm)

The Problem

The ability to change tuning parameters while the system is running is a must for any respectable PID algorithm.

技术分享图片

The Beginner’s PID acts a little crazy if you try to change the tunings while it’s running. Let’s see why. Here is the state of the beginner’s PID before and after the parameter change above:

技术分享图片

 

So we can immediately blame this bump on the Integral Term (or “I Term”). It’s the only thing that changes drastically when the parameters change. Why did this happen? It has to do with the beginner’s interpretation of the Integral:

技术分享图片

This interpretation works fine until the Ki is changed. Then, all of a sudden, you multiply this new Ki times the entire error sum that you have accumulated. That’s not what we wanted! We only wanted to affect things moving forward!

The Solution

There are a couple ways I know of to deal with this problem. The method I used in the last library was to rescale errSum. Ki doubled? Cut errSum in Half. That keeps the I Term from bumping, and it works. It’s kind of clunky though, and I’ve come up with something more elegant. (There’s no way I’m the first to have thought of this, but I did think of it on my own. That counts damnit!)

The solution requires a little basic algebra (or is it calculus?)

技术分享图片

Instead of having the Ki live outside the integral, we bring it inside. It looks like we haven’t done anything, but we’ll see that in practice this makes a big difference.

Now, we take the error and multiply it by whatever the Ki is at that time. We then store the sum of THAT. When the Ki changes, there’s no bump because all the old Ki’s are already “in the bank” so to speak. We get a smooth transfer with no additional math operations. It may make me a geek but I think that’s pretty sexy.

 1 /*working variables*/
 2 unsigned long lastTime;
 3 double Input, Output, Setpoint;
 4 double ITerm, lastInput;
 5 double kp, ki, kd;
 6 int SampleTime = 1000; //1 sec
 7 void Compute()
 8 {
 9    unsigned long now = millis();
10    int timeChange = (now - lastTime);
11    if(timeChange>=SampleTime)
12    {
13       /*Compute all the working error variables*/
14       double error = Setpoint - Input;
15       ITerm += (ki * error);
16       double dInput = (Input - lastInput);
17  
18       /*Compute PID Output*/
19       Output = kp * error + ITerm - kd * dInput;
20  
21       /*Remember some variables for next time*/
22       lastInput = Input;
23       lastTime = now;
24    }
25 }
26  
27 void SetTunings(double Kp, double Ki, double Kd)
28 {
29   double SampleTimeInSec = ((double)SampleTime)/1000;
30    kp = Kp;
31    ki = Ki * SampleTimeInSec;
32    kd = Kd / SampleTimeInSec;
33 }
34  
35 void SetSampleTime(int NewSampleTime)
36 {
37    if (NewSampleTime > 0)
38    {
39       double ratio  = (double)NewSampleTime
40                       / (double)SampleTime;
41       ki *= ratio;
42       kd /= ratio;
43       SampleTime = (unsigned long)NewSampleTime;
44    }
45 }

So we replaced the errSum variable with a composite ITerm variable [Line 4]. It sums Ki*error, rather than just error [Line 15]. Also, because Ki is now buried in ITerm, it’s removed from the main PID calculation [Line 19].

The Result

技术分享图片
技术分享图片
So how does this fix things. Before when ki was changed, it rescaled the entire sum of the error; every error value we had seen. With this code, the previous error remains untouched, and the new ki only affects things moving forward, which is exactly what we want.

 

Improving the Beginner’s PID: Reset Windup

(This is Modification #4 in a larger series on writing a solid PID algorithm)

The Problem

技术分享图片
Reset windup is a trap that probably claims more beginners than any other. It occurs when the PID thinks it can do something that it can’t. For example, the PWM output on an Arduino accepts values from 0-255. By default the PID doesn’t know this. If it thinks that 300-400-500 will work, it’s going to try those values expecting to get what it needs. Since in reality the value is clamped at 255 it’s just going to keep trying higher and higher numbers without getting anywhere.

The problem reveals itself in the form of weird lags. Above we can see that the output gets “wound up” WAY above the external limit. When the setpoint is dropped the output has to wind down before getting below that 255-line.

The Solution – Step 1

技术分享图片
There are several ways that windup can be mitigated, but the one that I chose was as follows: tell the PID what the output limits are. In the code below you’ll see there’s now a SetOuputLimits function. Once either limit is reached, the pid stops summing (integrating.) It knows there’s nothing to be done; Since the output doesn’t wind-up, we get an immediate response when the setpoint drops into a range where we can do something.

The Solution – Step 2

Notice in the graph above though, that while we got rid that windup lag, we’re not all the way there. There’s still a difference between what the pid thinks it’s sending, and what’s being sent. Why? the Proportional Term and (to a lesser extent) the Derivative Term.

Even though the Integral Term has been safely clamped, P and D are still adding their two cents, yielding a result higher than the output limit. To my mind this is unacceptable. If the user calls a function called “SetOutputLimits” they’ve got to assume that that means “the output will stay within these values.” So for Step 2, we make that a valid assumption. In addition to clamping the I-Term, we clamp the Output value so that it stays where we’d expect it.

(Note: You might ask why we need to clamp both. If we’re going to do the output anyway, why clamp the Integral separately? If all we did was clamp the output, the Integral term would go back to growing and growing. Though the output would look nice during the step up, we’d see that telltale lag on the step down.)

 
 1 /*working variables*/
 2 unsigned long lastTime;
 3 double Input, Output, Setpoint;
 4 double ITerm, lastInput;
 5 double kp, ki, kd;
 6 int SampleTime = 1000; //1 sec
 7 double outMin, outMax;
 8 void Compute()
 9 {
10    unsigned long now = millis();
11    int timeChange = (now - lastTime);
12    if(timeChange>=SampleTime)
13    {
14       /*Compute all the working error variables*/
15       double error = Setpoint - Input;
16       ITerm+= (ki * error);
17       if(ITerm> outMax) ITerm= outMax;
18       else if(ITerm< outMin) ITerm= outMin;
19       double dInput = (Input - lastInput);
20  
21       /*Compute PID Output*/
22       Output = kp * error + ITerm- kd * dInput;
23       if(Output > outMax) Output = outMax;
24       else if(Output < outMin) Output = outMin;
25  
26       /*Remember some variables for next time*/
27       lastInput = Input;
28       lastTime = now;
29    }
30 }
31  
32 void SetTunings(double Kp, double Ki, double Kd)
33 {
34   double SampleTimeInSec = ((double)SampleTime)/1000;
35    kp = Kp;
36    ki = Ki * SampleTimeInSec;
37    kd = Kd / SampleTimeInSec;
38 }
39  
40 void SetSampleTime(int NewSampleTime)
41 {
42    if (NewSampleTime > 0)
43    {
44       double ratio  = (double)NewSampleTime
45                       / (double)SampleTime;
46       ki *= ratio;
47       kd /= ratio;
48       SampleTime = (unsigned long)NewSampleTime;
49    }
50 }
51  
52 void SetOutputLimits(double Min, double Max)
53 {
54    if(Min > Max) return;
55    outMin = Min;
56    outMax = Max;
57     
58    if(Output > outMax) Output = outMax;
59    else if(Output < outMin) Output = outMin;
60  
61    if(ITerm> outMax) ITerm= outMax;
62    else if(ITerm< outMin) ITerm= outMin;
63 }

A new function was added to allow the user to specify the output limits [lines 52-63]. And these limits are used to clamp both the I-Term [17-18] and the Output [23-24]

The Result

技术分享图片
As we can see, windup is eliminated. in addition, the output stays where we want it to. this means there’s no need for external clamping of the output. if you want it to range from 23 to 167, you can set those as the Output Limits.

 

Improving the Beginner’s PID: On/Off

(This is Modification #5 in a larger series on writing a solid PID algorithm)

The Problem

As nice as it is to have a PID controller, sometimes you don’t care what it has to say.

技术分享图片
Let’s say at some point in your program you want to force the output to a certain value (0 for example) you could certainly do this in the calling routine:

void loop()
{
Compute();
Output=0;
}

This way, no matter what the PID says, you just overwrite its value. This is a terrible idea in practice however. The PID will become very confused: “I keep moving the output, and nothing’s happening! What gives?! Let me move it some more.” As a result, when you stop over-writing the output and switch back to the PID, you will likely get a huge and immediate change in the output value.

The Solution

The solution to this problem is to have a means to turn the PID off and on. The common terms for these states are “Manual” (I will adjust the value by hand) and “Automatic” (the PID will automatically adjust the output). Let’s see how this is done in code:

 1 /*working variables*/
 2 unsigned long lastTime;
 3 double Input, Output, Setpoint;
 4 double ITerm, lastInput;
 5 double kp, ki, kd;
 6 int SampleTime = 1000; //1 sec
 7 double outMin, outMax;
 8 bool inAuto = false;
 9  
10 #define MANUAL 0
11 #define AUTOMATIC 1
12  
13 void Compute()
14 {
15    if(!inAuto) return;
16    unsigned long now = millis();
17    int timeChange = (now - lastTime);
18    if(timeChange>=SampleTime)
19    {
20       /*Compute all the working error variables*/
21       double error = Setpoint - Input;
22       ITerm+= (ki * error);
23       if(ITerm> outMax) ITerm= outMax;
24       else if(ITerm< outMin) ITerm= outMin;
25       double dInput = (Input - lastInput);
26  
27       /*Compute PID Output*/
28       Output = kp * error + ITerm- kd * dInput;
29       if(Output > outMax) Output = outMax;
30       else if(Output < outMin) Output = outMin;
31  
32       /*Remember some variables for next time*/
33       lastInput = Input;
34       lastTime = now;
35    }
36 }
37  
38 void SetTunings(double Kp, double Ki, double Kd)
39 {
40   double SampleTimeInSec = ((double)SampleTime)/1000;
41    kp = Kp;
42    ki = Ki * SampleTimeInSec;
43    kd = Kd / SampleTimeInSec;
44 }
45  
46 void SetSampleTime(int NewSampleTime)
47 {
48    if (NewSampleTime > 0)
49    {
50       double ratio  = (double)NewSampleTime
51                       / (double)SampleTime;
52       ki *= ratio;
53       kd /= ratio;
54       SampleTime = (unsigned long)NewSampleTime;
55    }
56 }
57  
58 void SetOutputLimits(double Min, double Max)
59 {
60    if(Min > Max) return;
61    outMin = Min;
62    outMax = Max;
63     
64    if(Output > outMax) Output = outMax;
65    else if(Output < outMin) Output = outMin;
66  
67    if(ITerm> outMax) ITerm= outMax;
68    else if(ITerm< outMin) ITerm= outMin;
69 }
70  
71 void SetMode(int Mode)
72 {
73   inAuto = (Mode == AUTOMATIC);
74 }

A fairly simple solution. If you’re not in automatic mode, immediately leave the Compute function without adjusting the Output or any internal variables.

 

The Result

技术分享图片
It’s true that you could achieve a similar effect by just not calling Compute from the calling routine, but this solution keeps the workings of the PID contained, which is kind of what we need. By keeping things internal we can keep track of which mode were in, and more importantly it let’s us know when we change modes. That leads us to the next issue…

 

 

Improving the Beginner’s PID: Initialization

(This is Modification #6 in a larger series on writing a solid PID algorithm)

The Problem

In the last section we implemented the ability to turn the PID off and on. We turned it off, but now let’s look at what happens when we turn it back on:
技术分享图片

Yikes! The PID jumps back to the last Output value it sent, then starts adjusting from there. This results in an Input bump that we’d rather not have.

The Solution

This one is pretty easy to fix. Since we now know when we’re turning on (going from Manual to Automatic,) we just have to initialize things for a smooth transition. That means massaging the 2 stored working variables (ITerm & lastInput) to keep the output from jumping.

 1 /*working variables*/
 2 unsigned long lastTime;
 3 double Input, Output, Setpoint;
 4 double ITerm, lastInput;
 5 double kp, ki, kd;
 6 int SampleTime = 1000; //1 sec
 7 double outMin, outMax;
 8 bool inAuto = false;
 9  
10 #define MANUAL 0
11 #define AUTOMATIC 1
12  
13 void Compute()
14 {
15    if(!inAuto) return;
16    unsigned long now = millis();
17    int timeChange = (now - lastTime);
18    if(timeChange>=SampleTime)
19    {
20       /*Compute all the working error variables*/
21       double error = Setpoint - Input;
22       ITerm+= (ki * error);
23       if(ITerm> outMax) ITerm= outMax;
24       else if(ITerm< outMin) ITerm= outMin;
25       double dInput = (Input - lastInput);
26  
27       /*Compute PID Output*/
28       Output = kp * error + ITerm- kd * dInput;
29       if(Output> outMax) Output = outMax;
30       else if(Output < outMin) Output = outMin;
31  
32       /*Remember some variables for next time*/
33       lastInput = Input;
34       lastTime = now;
35    }
36 }
37  
38 void SetTunings(double Kp, double Ki, double Kd)
39 {
40   double SampleTimeInSec = ((double)SampleTime)/1000;
41    kp = Kp;
42    ki = Ki * SampleTimeInSec;
43    kd = Kd / SampleTimeInSec;
44 }
45  
46 void SetSampleTime(int NewSampleTime)
47 {
48    if (NewSampleTime > 0)
49    {
50       double ratio  = (double)NewSampleTime
51                       / (double)SampleTime;
52       ki *= ratio;
53       kd /= ratio;
54       SampleTime = (unsigned long)NewSampleTime;
55    }
56 }
57  
58 void SetOutputLimits(double Min, double Max)
59 {
60    if(Min > Max) return;
61    outMin = Min;
62    outMax = Max;
63     
64    if(Output > outMax) Output = outMax;
65    else if(Output < outMin) Output = outMin;
66  
67    if(ITerm> outMax) ITerm= outMax;
68    else if(ITerm< outMin) ITerm= outMin;
69 }
70  
71 void SetMode(int Mode)
72 {
73     bool newAuto = (Mode == AUTOMATIC);
74     if(newAuto && !inAuto)
75     {  /*we just went from manual to auto*/
76         Initialize();
77     }
78     inAuto = newAuto;
79 }
80  
81 void Initialize()
82 {
83    lastInput = Input;
84    ITerm = Output;
85    if(ITerm> outMax) ITerm= outMax;
86    else if(ITerm< outMin) ITerm= outMin;
87 }

We modified SetMode(…) to detect the transition from manual to automatic, and we added our initialization function. It sets ITerm=Output to take care of the integral term, and lastInput = Input to keep the derivative from spiking. The proportional term doesn’t rely on any information from the past, so it doesn’t need any initialization.

The Result

技术分享图片

We see from the above graph that proper initialization results in a bumpless transfer from manual to automatic: exactly what we were after.

Update: Why not ITerm=0?

I have been getting a lot of questions recently asking why I don’t set ITerm=0 upon intialization. As an answer, I’d ask you to consider the following scenario: The pid is in manual, and the user has set the output to 50. After a time, the process steadies out to an input of 75.2. The user makes the Setpoint 75.2 and turns on the pid. What should happen?

I contend that after switching to automatic the output value should stay at 50. since the P and D terms will be zero, the only way this will happen is if ITerm is initialized to the value of Output.

If you are in a situation where you need the output to initialize to zero, there is no need alter the code above. Just set Output=0 in your calling routine before turning the PID from Manual to Automatic.

 

 

Improving the Beginner’s PID: Direction

(This is the last modification in a larger series on writing a solid PID algorithm)

The Problem

The processes the PID will be connected to fall into two groups: direct acting and reverse acting. All the examples I’ve shown so far have been direct acting. That is, an increase in the output causes an increase in the input. For reverse acting processes the opposite is true. In a refrigerator for example, an increase in cooling causes the temperature to go down. To make the beginner PID work with a reverse process, the signs of kp, ki, and kd all must be negative.

This isn’t a problem per se, but the user must choose the correct sign, and make sure that all the parameters have the same sign.

The Solution

To make the process a little simpler, I require that kp, ki, and kd all be >=0. If the user is connected to a reverse process, they specify that separately using the SetControllerDirection function. this ensures that the parameters all have the same sign, and hopefully makes things more intuitive.

  1 /*working variables*/
  2 unsigned long lastTime;
  3 double Input, Output, Setpoint;
  4 double ITerm, lastInput;
  5 double kp, ki, kd;
  6 int SampleTime = 1000; //1 sec
  7 double outMin, outMax;
  8 bool inAuto = false;
  9  
 10 #define MANUAL 0
 11 #define AUTOMATIC 1
 12  
 13 #define DIRECT 0
 14 #define REVERSE 1
 15 int controllerDirection = DIRECT;
 16  
 17 void Compute()
 18 {
 19    if(!inAuto) return;
 20    unsigned long now = millis();
 21    int timeChange = (now - lastTime);
 22    if(timeChange>=SampleTime)
 23    {
 24       /*Compute all the working error variables*/
 25       double error = Setpoint - Input;
 26       ITerm+= (ki * error);
 27       if(ITerm > outMax) ITerm= outMax;
 28       else if(ITerm < outMin) ITerm= outMin;
 29       double dInput = (Input - lastInput);
 30  
 31       /*Compute PID Output*/
 32       Output = kp * error + ITerm- kd * dInput;
 33       if(Output > outMax) Output = outMax;
 34       else if(Output < outMin) Output = outMin;
 35  
 36       /*Remember some variables for next time*/
 37       lastInput = Input;
 38       lastTime = now;
 39    }
 40 }
 41  
 42 void SetTunings(double Kp, double Ki, double Kd)
 43 {
 44    if (Kp<0 || Ki<0|| Kd<0) return;
 45  
 46   double SampleTimeInSec = ((double)SampleTime)/1000;
 47    kp = Kp;
 48    ki = Ki * SampleTimeInSec;
 49    kd = Kd / SampleTimeInSec;
 50  
 51   if(controllerDirection ==REVERSE)
 52    {
 53       kp = (0 - kp);
 54       ki = (0 - ki);
 55       kd = (0 - kd);
 56    }
 57 }
 58  
 59 void SetSampleTime(int NewSampleTime)
 60 {
 61    if (NewSampleTime > 0)
 62    {
 63       double ratio  = (double)NewSampleTime
 64                       / (double)SampleTime;
 65       ki *= ratio;
 66       kd /= ratio;
 67       SampleTime = (unsigned long)NewSampleTime;
 68    }
 69 }
 70  
 71 void SetOutputLimits(double Min, double Max)
 72 {
 73    if(Min > Max) return;
 74    outMin = Min;
 75    outMax = Max;
 76  
 77    if(Output > outMax) Output = outMax;
 78    else if(Output < outMin) Output = outMin;
 79  
 80    if(ITerm > outMax) ITerm= outMax;
 81    else if(ITerm < outMin) ITerm= outMin;
 82 }
 83  
 84 void SetMode(int Mode)
 85 {
 86     bool newAuto = (Mode == AUTOMATIC);
 87     if(newAuto == !inAuto)
 88     {  /*we just went from manual to auto*/
 89         Initialize();
 90     }
 91     inAuto = newAuto;
 92 }
 93  
 94 void Initialize()
 95 {
 96    lastInput = Input;
 97    ITerm = Output;
 98    if(ITerm > outMax) ITerm= outMax;
 99    else if(ITerm < outMin) ITerm= outMin;
100 }
101  
102 void SetControllerDirection(int Direction)
103 {
104    controllerDirection = Direction;
105 }

 

PID COMPLETE

And that about wraps it up. We’ve turned “The Beginner’s PID” into the most robust controller I know how to make at this time. For those readers that were looking for a detailed explanation of the PID Library, I hope you got what you came for. For those of you writing your own PID, I hope you were able to glean a few ideas that save you some cycles down the road.

Two Final Notes:

  1. If something in this series looks wrong please let me know. I may have missed something, or might just need to be clearer in my explanation. Either way I’d like to know.
  2. This is just a basic PID. There are many other issues that I intentionally left out in the name of simplicity. Off the top of my head: feed forward, reset tiebacks, integer math, different pid forms, using velocity instead of position. If there’s interest in having me explore these topics please let me know.

 

Proportional on Measurement – The Code

In the previous post I spent all my time explaining the benefits of Proportional on Measurement. In this post I’ll explain the code. People seemed to appreciate the step-by-step way I explained things last time, so that’s what I’ll do here. The 3 passes below detail how I went about adding PonM to the PID library.

First Pass – Initial Input and Proportional-Mode selection

  1 /*working variables*/
  2 unsigned long lastTime;
  3 double Input, Output, Setpoint;
  4 double ITerm, lastInput;
  5 double kp, ki, kd;
  6 int SampleTime = 1000; //1 sec
  7 double outMin, outMax;
  8 bool inAuto = false;
  9   
 10 #define MANUAL 0
 11 #define AUTOMATIC 1
 12   
 13 #define DIRECT 0
 14 #define REVERSE 1
 15 int controllerDirection = DIRECT;
 16   
 17 #define P_ON_M 0
 18 #define P_ON_E 1
 19 bool pOnE = true;
 20 double initInput;
 21  
 22 void Compute()
 23 {
 24    if(!inAuto) return;
 25    unsigned long now = millis();
 26    int timeChange = (now - lastTime);
 27    if(timeChange>=SampleTime)
 28    {
 29       /*Compute all the working error variables*/
 30       double error = Setpoint - Input;
 31       ITerm+= (ki * error);
 32       if(ITerm > outMax) ITerm= outMax;
 33       else if(ITerm < outMin) ITerm= outMin;
 34       double dInput = (Input - lastInput);
 35   
 36       /*Compute P-Term*/
 37       if(pOnE) Output = kp * error;
 38       else Output = -kp * (Input-initInput); 
 39  
 40       /*Compute Rest of PID Output*/
 41       Output += ITerm - kd * dInput;
 42       if(Output > outMax) Output = outMax;
 43       else if(Output < outMin) Output = outMin;
 44   
 45       /*Remember some variables for next time*/
 46       lastInput = Input;
 47       lastTime = now;
 48    }
 49 }
 50   
 51 void SetTunings(double Kp, double Ki, double Kd, int pOn)
 52 {
 53    if (Kp<0 || Ki<0|| Kd<0) return;
 54   
 55    pOnE = pOn == P_ON_E;
 56    
 57    double SampleTimeInSec = ((double)SampleTime)/1000;
 58    kp = Kp;
 59    ki = Ki * SampleTimeInSec;
 60    kd = Kd / SampleTimeInSec;
 61   
 62   if(controllerDirection ==REVERSE)
 63    {
 64       kp = (0 - kp);
 65       ki = (0 - ki);
 66       kd = (0 - kd);
 67    }
 68 }
 69   
 70 void SetSampleTime(int NewSampleTime)
 71 {
 72    if (NewSampleTime > 0)
 73    {
 74       double ratio  = (double)NewSampleTime
 75                       / (double)SampleTime;
 76       ki *= ratio;
 77       kd /= ratio;
 78       SampleTime = (unsigned long)NewSampleTime;
 79    }
 80 }
 81   
 82 void SetOutputLimits(double Min, double Max)
 83 {
 84    if(Min > Max) return;
 85    outMin = Min;
 86    outMax = Max;
 87   
 88    if(Output > outMax) Output = outMax;
 89    else if(Output < outMin) Output = outMin;
 90   
 91    if(ITerm > outMax) ITerm= outMax;
 92    else if(ITerm < outMin) ITerm= outMin;
 93 }
 94   
 95 void SetMode(int Mode)
 96 {
 97     bool newAuto = (Mode == AUTOMATIC);
 98     if(newAuto == !inAuto)
 99     {  /*we just went from manual to auto*/
100         Initialize();
101     }
102     inAuto = newAuto;
103 }
104   
105 void Initialize()
106 {
107    lastInput = Input;
108    initInput = Input;
109    ITerm = Output;
110    if(ITerm > outMax) ITerm= outMax;
111    else if(ITerm < outMin) ITerm= outMin;
112 }
113   
114 void SetControllerDirection(int Direction)
115 {
116    controllerDirection = Direction;
117 }

 

Proportional on Measurement provides increasing resistance as the input changes, but without a frame of reference our performance would be a little wonky. If the PID Input is 10000 when we first turn on the controller, do we really want to start resisting with Kp*10000? No. We want to use our initial input as a reference point (line 108,) the increase or decrease resistance as the input changes from there (line 38.)

The other thing we need to do is allow the user to select whether they want to do Proportional on Error or Measurement. After the last post it might seem like PonE is useless, but it’s important to remember that for many loops it works well. As such, we need to let the user choose which mode they want (lines 51&55) and then act accordingly in the calculation (lines 37&38).

Second Pass – On-the-fly tuning changes

While the code above does indeed work, it has a problem that we’ve seen before. When tuning parameters are changed at runtime, we get an undesirable blip.

技术分享图片

Why is this happening?

技术分享图片

The last time we saw this, it was that the integral was being rescaled by a new Ki. This time, it’s that (Input – initInput) is being rescaled by Kp. the solution I chose is similar to what I did for Ki: instead of treating the Input – initInput as a monolithic unit multiplied by the current Kp, I broke it into individual steps multiplied by the Kp at that time:

技术分享图片

 

  1 /*working variables*/
  2 unsigned long lastTime;
  3 double Input, Output, Setpoint;
  4 double ITerm, lastInput;
  5 double kp, ki, kd;
  6 int SampleTime = 1000; //1 sec
  7 double outMin, outMax;
  8 bool inAuto = false;
  9   
 10 #define MANUAL 0
 11 #define AUTOMATIC 1
 12   
 13 #define DIRECT 0
 14 #define REVERSE 1
 15 int controllerDirection = DIRECT;
 16   
 17 #define P_ON_M 0
 18 #define P_ON_E 1
 19 bool pOnE = true;
 20 double PTerm;
 21  
 22 void Compute()
 23 {
 24    if(!inAuto) return;
 25    unsigned long now = millis();
 26    int timeChange = (now - lastTime);
 27    if(timeChange>=SampleTime)
 28    {
 29     
 30       /*Compute all the working error variables*/      
 31       double error = Setpoint - Input;   
 32       ITerm+= (ki * error);  
 33       if(ITerm > outMax) ITerm= outMax;      
 34       else if(ITerm < outMin) ITerm= outMin;  
 35       double dInput = (Input - lastInput);
 36  
 37       /*Compute P-Term*/
 38       if(pOnE) Output = kp * error; 
 39       else 
 40       { 
 41          PTerm -= kp * dInput; 
 42          Output = PTerm; 
 43       } 
 44        
 45       /*Compute Rest of PID Output*/
 46       Output += ITerm - kd * dInput; 
 47     
 48       if(Output > outMax) Output = outMax;
 49       else if(Output < outMin) Output = outMin;
 50   
 51       /*Remember some variables for next time*/
 52       lastInput = Input;
 53       lastTime = now;
 54    }
 55 }
 56   
 57 void SetTunings(double Kp, double Ki, double Kd, int pOn)
 58 {
 59    if (Kp<0 || Ki<0|| Kd<0) return;
 60   
 61    pOnE = pOn == P_ON_E;
 62    
 63    double SampleTimeInSec = ((double)SampleTime)/1000;
 64    kp = Kp;
 65    ki = Ki * SampleTimeInSec;
 66    kd = Kd / SampleTimeInSec;
 67   
 68   if(controllerDirection ==REVERSE)
 69    {
 70       kp = (0 - kp);
 71       ki = (0 - ki);
 72       kd = (0 - kd);
 73    }
 74 }
 75   
 76 void SetSampleTime(int NewSampleTime)
 77 {
 78    if (NewSampleTime > 0)
 79    {
 80       double ratio  = (double)NewSampleTime
 81                       / (double)SampleTime;
 82       ki *= ratio;
 83       kd /= ratio;
 84       SampleTime = (unsigned long)NewSampleTime;
 85    }
 86 }
 87   
 88 void SetOutputLimits(double Min, double Max)
 89 {
 90    if(Min > Max) return;
 91    outMin = Min;
 92    outMax = Max;
 93   
 94    if(Output > outMax) Output = outMax;
 95    else if(Output < outMin) Output = outMin;
 96   
 97    if(ITerm > outMax) ITerm= outMax;
 98    else if(ITerm < outMin) ITerm= outMin;
 99 }
100   
101 void SetMode(int Mode)
102 {
103     bool newAuto = (Mode == AUTOMATIC);
104     if(newAuto == !inAuto)
105     {  /*we just went from manual to auto*/
106         Initialize();
107     }
108     inAuto = newAuto;
109 }
110   
111 void Initialize()
112 {
113    lastInput = Input;
114    PTerm = 0;
115    ITerm = Output;
116    if(ITerm > outMax) ITerm= outMax;
117    else if(ITerm < outMin) ITerm= outMin;
118 }
119   
120 void SetControllerDirection(int Direction)
121 {
122    controllerDirection = Direction;
123 }

 

Instead of multiplying the entirety of Input-initInput by Kp, we now keep a working sum, PTerm. At each step we multiply just the current input change by Kp and subtract it from PTerm (line 41.) Here we can see the impact of the change:

技术分享图片

技术分享图片

Because the old Kps are “in the bank”, the change in tuning parameters only affects us moving forward

Final Pass – Sum problems.

I won’t go into complete detail (fancy trends etc) as to what wrong with the above code. It’s pretty good, but there are still major issues with it. For example:

  1. Windup, sort of: While the final output is limited between outMin and outMax, it’s possible for PTerm to grow when it shouldn’t. It wouldn’t be as bad as integral windup, but it still wouldn’t be acceptable
  2. On-the-fly changes: If the user were to change from P_ON_M to P_ON_E while running, then after some time return back, PTerm wouldn’t be property initialized and it would cause an output bump

There are more, but just these are enough to see what the real issue is. We’ve dealt with all of these before, back when we created ITerm. Rather than go through and re-implement the same solutions for PTerm, I decided on a more aesthetically-pleasing solution.

By merging PTerm and ITerm into a single variable called “outputSum”, the P_ON_M code then benefits from the all the ITerm fixes that are already in place, and because there aren’t two sums in the code there isn’t needless redundancy.

 
  1 /*working variables*/
  2 unsigned long lastTime;
  3 double Input, Output, Setpoint;
  4 double outputSum, lastInput;
  5 double kp, ki, kd;
  6 int SampleTime = 1000; //1 sec
  7 double outMin, outMax;
  8 bool inAuto = false;
  9   
 10 #define MANUAL 0
 11 #define AUTOMATIC 1
 12   
 13 #define DIRECT 0
 14 #define REVERSE 1
 15 int controllerDirection = DIRECT;
 16   
 17 #define P_ON_M 0
 18 #define P_ON_E 1
 19 bool pOnE = true;
 20  
 21  
 22 void Compute()
 23 {
 24    if(!inAuto) return;
 25    unsigned long now = millis();
 26    int timeChange = (now - lastTime);
 27    if(timeChange>=SampleTime)
 28    {
 29     
 30       /*Compute all the working error variables*/      
 31       double error = Setpoint - Input;   
 32       double dInput = (Input - lastInput);
 33       outputSum+= (ki * error);  
 34        
 35       /*Add Proportional on Measurement, if P_ON_M is specified*/
 36       if(!pOnE) outputSum-= kp * dInput
 37        
 38       if(outputSum > outMax) outputSum= outMax;      
 39       else if(outputSum < outMin) outputSum= outMin;  
 40      
 41       /*Add Proportional on Error, if P_ON_E is specified*/
 42       if(pOnE) Output = kp * error; 
 43       else Output = 0;
 44        
 45       /*Compute Rest of PID Output*/
 46       Output += outputSum - kd * dInput; 
 47     
 48       if(Output > outMax) Output = outMax;
 49       else if(Output < outMin) Output = outMin;
 50   
 51       /*Remember some variables for next time*/
 52       lastInput = Input;
 53       lastTime = now;
 54    }
 55 }
 56   
 57 void SetTunings(double Kp, double Ki, double Kd, int pOn)
 58 {
 59    if (Kp<0 || Ki<0|| Kd<0) return;
 60   
 61    pOnE = pOn == P_ON_E;
 62    
 63    double SampleTimeInSec = ((double)SampleTime)/1000;
 64    kp = Kp;
 65    ki = Ki * SampleTimeInSec;
 66    kd = Kd / SampleTimeInSec;
 67   
 68   if(controllerDirection ==REVERSE)
 69    {
 70       kp = (0 - kp);
 71       ki = (0 - ki);
 72       kd = (0 - kd);
 73    }
 74 }
 75   
 76 void SetSampleTime(int NewSampleTime)
 77 {
 78    if (NewSampleTime > 0)
 79    {
 80       double ratio  = (double)NewSampleTime
 81                       / (double)SampleTime;
 82       ki *= ratio;
 83       kd /= ratio;
 84       SampleTime = (unsigned long)NewSampleTime;
 85    }
 86 }
 87   
 88 void SetOutputLimits(double Min, double Max)
 89 {
 90    if(Min > Max) return;
 91    outMin = Min;
 92    outMax = Max;
 93   
 94    if(Output > outMax) Output = outMax;
 95    else if(Output < outMin) Output = outMin;
 96   
 97    if(outputSum > outMax) outputSum= outMax;
 98    else if(outputSum < outMin) outputSum= outMin;
 99 }
100   
101 void SetMode(int Mode)
102 {
103     bool newAuto = (Mode == AUTOMATIC);
104     if(newAuto == !inAuto)
105     {  /*we just went from manual to auto*/
106         Initialize();
107     }
108     inAuto = newAuto;
109 }
110   
111 void Initialize()
112 {
113    lastInput = Input;
114     
115    outputSum = Output;
116    if(outputSum > outMax) outputSum= outMax;
117    else if(outputSum < outMin) outputSum= outMin;
118 }
119   
120 void SetControllerDirection(int Direction)
121 {
122    controllerDirection = Direction;
123 }

 

And there you have it. The above functionality is what is now present in v1.2.0 of the Arduino PID.

But wait, there’s more: Setpoint Weighting.

I didn’t add the following to the Arduino library code, but this is a feature that might be of some interest if you want to roll your own. Setpoint Weighting is, at it’s core, a way to have both PonE and PonM at the same time. By specifying a ratio between 0 and 1, you can have 100% PonM, 100% PonE (respectively) or some ratio in between. This can be helpful if you have a process that’s not perfectly integrating (like a reflow oven) and want to account for this.

Ultimately I decided not to add it to the library at this time, as it winds up being ANOTHER parameter to adjust/explain, and I didn’t think the resulting benefit was worth it. At any rate, here is the code if you’d like to modify the code to have Setpoint Weighting instead of just pure PonM/PonE selection:

  1 /*working variables*/
  2 unsigned long lastTime;
  3 double Input, Output, Setpoint;
  4 double outputSum, lastInput;
  5 double kp, ki, kd;
  6 int SampleTime = 1000; //1 sec
  7 double outMin, outMax;
  8 bool inAuto = false;
  9   
 10 #define MANUAL 0
 11 #define AUTOMATIC 1
 12   
 13 #define DIRECT 0
 14 #define REVERSE 1
 15 int controllerDirection = DIRECT;
 16   
 17 #define P_ON_M 0
 18 #define P_ON_E 1
 19 bool pOnE = true, pOnM = false;
 20 double pOnEKp, pOnMKp;
 21  
 22  
 23 void Compute()
 24 {
 25    if(!inAuto) return;
 26    unsigned long now = millis();
 27    int timeChange = (now - lastTime);
 28    if(timeChange>=SampleTime)
 29    {
 30     
 31       /*Compute all the working error variables*/      
 32       double error = Setpoint - Input;   
 33       double dInput = (Input - lastInput);
 34       outputSum+= (ki * error);  
 35        
 36       /*Add Proportional on Measurement, if P_ON_M is specified*/
 37       if(pOnM) outputSum-= pOnMKp * dInput
 38        
 39       if(outputSum > outMax) outputSum= outMax;      
 40       else if(outputSum < outMin) outputSum= outMin;  
 41      
 42       /*Add Proportional on Error, if P_ON_E is specified*/
 43       if(pOnE) Output = pOnEKp * error; 
 44       else Output = 0;
 45        
 46       /*Compute Rest of PID Output*/
 47       Output += outputSum - kd * dInput; 
 48     
 49       if(Output > outMax) Output = outMax;
 50       else if(Output < outMin) Output = outMin;
 51   
 52       /*Remember some variables for next time*/
 53       lastInput = Input;
 54       lastTime = now;
 55    }
 56 }
 57   
 58 void SetTunings(double Kp, double Ki, double Kd, double pOn)
 59 {
 60    if (Kp<0 || Ki<0|| Kd<0 || pOn<0 || pOn>1) return;
 61   
 62    pOnE = pOn>0; //some p on error is desired;
 63    pOnM = pOn<1; //some p on measurement is desired;  
 64    
 65    double SampleTimeInSec = ((double)SampleTime)/1000;
 66    kp = Kp;
 67    ki = Ki * SampleTimeInSec;
 68    kd = Kd / SampleTimeInSec;
 69   
 70   if(controllerDirection ==REVERSE)
 71    {
 72       kp = (0 - kp);
 73       ki = (0 - ki);
 74       kd = (0 - kd);
 75    }
 76     
 77    pOnEKp = pOn * kp; 
 78    pOnMKp = (1 - pOn) * kp;
 79 }
 80   
 81 void SetSampleTime(int NewSampleTime)
 82 {
 83    if (NewSampleTime > 0)
 84    {
 85       double ratio  = (double)NewSampleTime
 86                       / (double)SampleTime;
 87       ki *= ratio;
 88       kd /= ratio;
 89       SampleTime = (unsigned long)NewSampleTime;
 90    }
 91 }
 92   
 93 void SetOutputLimits(double Min, double Max)
 94 {
 95    if(Min > Max) return;
 96    outMin = Min;
 97    outMax = Max;
 98   
 99    if(Output > outMax) Output = outMax;
100    else if(Output < outMin) Output = outMin;
101   
102    if(outputSum > outMax) outputSum= outMax;
103    else if(outputSum < outMin) outputSum= outMin;
104 }
105   
106 void SetMode(int Mode)
107 {
108     bool newAuto = (Mode == AUTOMATIC);
109     if(newAuto == !inAuto)
110     {  /*we just went from manual to auto*/
111         Initialize();
112     }
113     inAuto = newAuto;
114 }
115   
116 void Initialize()
117 {
118    lastInput = Input;
119    outputSum = Output;
120    if(outputSum > outMax) outputSum= outMax;
121    else if(outputSum < outMin) outputSum= outMin;
122 }
123   
124 void SetControllerDirection(int Direction)
125 {
126    controllerDirection = Direction;
127 }

 

Instead of setting pOn as an integer, it now comes in as a double which allows for a ratio (line 58.) In addition to some flags (lines 62&63) weighted Kp terms are computed on lines 77-78. Then on lines 37 and 43 the weighted PonM and PonE contributions are added to the overall PID output.

 

 

 

Brett Beauregard大神的Arduino PID算法

原文:https://www.cnblogs.com/FZLGYZ/p/11696323.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!