Introduction: The Role of PID in Feedback Control Systems
Proportional-Integral-Derivative (PID) controllers are among the most widely used control systems in engineering today. They are especially effective in single-input, single-output (SISO) applications where continuous feedback is required to maintain a desired output.
Given a plant process P that maps control actuator 




A classic example is the basic cruise control system you might find on your car. The control input signal is from the vehicle’s speed from the speedometer. The control output signal is the throttle position. From time to time, the car may encounter hills or other disturbances that necessitate a time-varying throttle output to maintain the speed reference.
Extending the Problem: Constrained SISO Systems
Given a plant process P, and system actions 





For example, maybe your car is actually a racecar, and there’s a torque limit on the tires that the engine should not exceed. In that case, the cruise control system needs to take the torque into account when setting the throttle, and there may be some situations, such as during an initial acceleration, when a normal SISO control loop wants to select throttles that would make the car exceed the torque specification.
In such a constrained-SISO problem there are three possible operating environments:
- The limitation is not a factor: . In that case, we want the controller to perform like a pure SISO controller on . 
- The limitation dominates the controller, so that it is not possible to get close to . In that case, we want the controller to perform like a pure SISO controller on , with . 
- In transitions between these two regimes, we want a smooth hand-off from the dynamics of one controller to the other.
In such a constrained-SISO problem there is a natural inclination to want to use PID feedback control to handle both stages. After all, PID is intuitive and ubiquitous in industry. The question is: How do you combine two feedback loops that both have control of only a single output signal? Many common approaches to the “merge problem” result in undesirable modal behaviors—where there are abrupt transients “handing off” from one loop to the other.
One Solution Approach
Here is one solution, that unfortunately I cannot take credit for:
The intuition here is as follows:  We want to control the 




But what is the upper limit on 








First Solution Analysis
Looking back at Figure 1, there is an algebraic loop, where the output signal 

Like any good Simulink problem, we can resolve this by adding a one-frame delay block to break the loop, and it turns out that this is the key to why this solution works.
To see why, let’s work out what an actual discrete-time controller is doing here[2]. The output saturation block works like this,
And the limit output 
After some algebra we arrive at the following two conclusions.
| When the main loop dominates the limiter loop… | When the limiter loop dominates the main loop… | 
| ![Rendered by QuickLaTeX.com u_{\text{PID}}[k] \leq u_{\text{lim}}[k]](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-69c6e0b33294beae71e4bdb686bc61c4_l3.png) | ![Rendered by QuickLaTeX.com u_{\text{PID}}[k] > u_{\text{lim}}[k]](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-0bd902092c7dd7825b3ca232a3ea671d_l3.png) | 
| ![Rendered by QuickLaTeX.com u[k+1] - u[k] \leq k_{\text{lim}} \left( x_2[k+1] - x_{2\text{lim}} \right)](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-dd056897dcd6f2dae6c4eec5418242bc_l3.png) | ![Rendered by QuickLaTeX.com u[k+1] - u[k] = k_{\text{lim}} \left( x_2[k+1] - x_{2\text{lim}} \right)](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-9397db5cb5513b3f83b717d5fd8c04d6_l3.png) | 
From this, we can see that the 




We can also use this result to analyze the controller in the continuous-time Laplace domain. When the limiter loop is dominating the main loop, we have
And after taking the Laplace transform (neglecting the constant):
Which rearranges to,
This has the form of an integral-only (I) controller, where the 

But integral-only control is not appropriate for many plant systems. Can we operate a PI controller instead?
To do so, we want to modify the transfer function so that it reads:
But from the block diagram, above we need a 

Neglecting the 

So
So to run PI control on the 
We can also reach a more general conclusion from all these Laplace transforms: The effect of breaking the algebraic loop using a delay is to apply one integrator to all control terms. Derivative terms become proportional terms; proportional terms become integral terms; and integral terms would become double-integral terms.
Can we redraw the block diagram so that we can have PI control on the  signal in the limiting case, without computing a derivative?
 signal in the limiting case, without computing a derivative?
Let’s try it.  The simplest, naïve way to get PI control on the main 

The question is: Does the merge using a min operation introduce undesirable modal switching behavior? That is, when we’re handing off from one loop to the other, do we get sudden and abrupt changes in the output signal?
To analyze this let’s look at the specific case where the main 



On the first frame, the control output is given by
Meanwhile, on the next frame the control output is given by
      
Now recall the same scenario with the original rate-limiting control scheme.  In the same situation where the 

We can line up these two equations.  Remember that on the second equation we effectively have 

That is, the integrator state for the 

Instead of computing the absolute ![Rendered by QuickLaTeX.com u[k]](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-28e9288f70054c98f60f2cb2601df1b8_l3.png)
![Rendered by QuickLaTeX.com u[k+1] - u[k]](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-41ff99f9d64c370a1cbf1e04507c271c_l3.png)
Notice that this is again equivalent to computing the derivative of the input signal.  In particular the 
![Rendered by QuickLaTeX.com k_p \bigl(\epsilon[k+1] - \epsilon[k]\bigr) \approx k_p \cdot \delta t \cdot \dot{\epsilon}(t)](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-6eef228ba63fc384db96a41dbe5cb30e_l3.png)
However, we do get one major benefit from implementing the discrete-time controller in this manner.  The “derivatives” that we do compute are pre-scaled by the discrete time period and by the 
In Conclusion…
Engineering teams, including Sparx Engineering, continue to select PID controllers to solve control problems because they are easy to use, intuitive to design, and they work well on a wide variety of problems.
In this post, we looked at two non-linear extensions to the PID architecture to solve the constrained single-loop control problem. This is a common situation that Sparx has had to deal with in diverse problem domains. In both extensions, we found that individual P-, I-, and D- terms are effectively integrated one level into I-, II-, and P- terms. This effect can be compensated by differentiating all controller terms, but beware of complications that may arise from computing instantaneous derivatives.
[1] In the cruise control example, the throttle valve can never be more than 100% open or less than 0% open.
[2] For simplicity, I will set 
[3] When 
![Rendered by QuickLaTeX.com k_{\text{lim}} \left(x_2[k+1] - x_{2\text{lim}}\right)](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-71a9d3f66cea3cbcd4146c21f1c7ce4c_l3.png)

[4] Once again, I am leaving out several features that would be in a real control law, like output saturation and anti-windup protection.
Featured Image Credit: https://benchmarkpdm.com/importance-of-shaft-alignment-parallel-angular-pump-alignment/
 . In that case, we want the controller to perform like a pure SISO controller on
. In that case, we want the controller to perform like a pure SISO controller on  . In that case, we want the controller to perform like a pure SISO controller on
. In that case, we want the controller to perform like a pure SISO controller on ![Rendered by QuickLaTeX.com u[k+1] = \begin{cases} u_{\text{PID}}[k+1], & u_{\text{PID}}[k+1] \leq u_{\text{lim}}[k] \\ u_{\text{lim}}[k+1], & u_{\text{PID}}[k+1] > u_{\text{lim}}[k] \end{cases}](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-c663c224d99c99931d06cde5015520a2_l3.png)
![Rendered by QuickLaTeX.com u_{\text{lim}}[k+1] = u[k] + k_{\text{lim}} \left( x_2[k+1] - x_{2\text{lim}} \right)](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-9bd12234e916ec22349d26a7c4a1e37b_l3.png)
![Rendered by QuickLaTeX.com \dot{u}(t) \approx \frac{1}{\delta t} \bigl(u[k+1] - u[k]\bigr) \leq \frac{k_{\text{lim}}}{\delta t} \left( x_2[k+1] - x_{2\text{lim}} \right)](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-11c79bc9f7a7bf41fd66ae7b76ba3351_l3.png)






![Rendered by QuickLaTeX.com I_{x_1}[k_0] = I_{x_1}[k_0 - 1] + \delta t \cdot k_i \left(x_1[k_0] - x_{1\text{ref}}\right)](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-06d1c7e0cb6e601ca4183766a9191191_l3.png)
![Rendered by QuickLaTeX.com u[k_0] = u_{x_1}[k_0] = k_p \left(x_1[k_0] - x_{1\text{ref}}\right) + I_{x_1}[k_0]](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-c7b9f19587385e6630182d20ca4a3f1b_l3.png)
![Rendered by QuickLaTeX.com \begin{align*}     I_{x_2}[k_0 + 1] &= I_{x_2}[k_0] + \delta t \cdot k_{i_{lim}}\left(x_2[k_0 + 1] - x_{2lim}\right) \\     u[k_0 + 1] = u_{x_2}[k_0 + 1] &= k_{p_{lim}}\left(x_2[k_0 + 1] - x_{2lim}\right) + I_{x_2}[k_0 + 1] \\     u[k_0 + 1] &= \left(k_{p_{lim}} + \delta t \cdot k_{i_{lim}}\right)\left(x_2[k_0 + 1] - x_{2lim}\right) + I_{x_2}[k_0]. \end{align*}](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-6a9aa449b74062e9485799d57c74db4a_l3.png)
![Rendered by QuickLaTeX.com u[k_0 + 1] - u[k_0] = k_{\text{lim}} \left(x_2[k_0 + 1] - x_{2\text{lim}}\right)](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-66158f8420b891142c585180e92d17dd_l3.png)
![Rendered by QuickLaTeX.com u[k_0 + 1] = k_{\text{lim}} \left(x_2[k_0 + 1] - x_{2\text{lim}}\right) + u[k_0]](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-dd37cf4bd6aa7b50fa15e64f5add8d32_l3.png)
![Rendered by QuickLaTeX.com I_{x_2}[k_0] = u[k_0] = u_{x_1}[k_0]](https://sparxeng.com/wp-content/ql-cache/quicklatex.com-375f63e191f83e3428def94e15c9057e_l3.png)