Oving-The-Beginners-Pid-Introduction/ Improving The Beginner's PID - Introduction
Oving-The-Beginners-Pid-Introduction/ Improving The Beginner's PID - Introduction
com/blog/2011/04/impr
oving-the-beginners-pid-introduction/
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.
This leads pretty much everyone to write the following PID controller:
1 /*working variables*/
2 unsigned long lastTime;
double Input, Output, Setpoint;
3 double errSum, lastErr;
4 double kp, ki, kd;
5 void Compute()
6 {
7 /*How long since we last calculated*/
unsigned long now = millis();
8 double timeChange = (double)(now -
9 lastTime);
10
11 /*Compute all the working error
12 variables*/
double error = Setpoint - Input;
13 errSum += (error * timeChange);
14 double dErr = (error - lastErr) /
15 timeChange;
16
17 /*Compute PID Output*/
18 Output = kp * error + ki * errSum + kd *
19 dErr;
20
/*Remember some variables for next
21 time*/
22 lastErr = error;
23 lastTime = now;
24 }
25
26 void SetTunings(double Kp, double Ki, double
Kd)
27 {
28 kp = Kp;
29 ki = Ki;
30 kd = Kd;
}
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
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
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
6. Initialization - When the controller first turns on, we want a “bumpless transfer.” That is, we don’t want the
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.
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.
Next >>
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