Control Theory Python Summary
Control Theory Python Summary
Walid El Bouhali
This Notebook contains all the important python commands that were used in the e-lectures
1 Transfer functions
[3]: # Tranfer functions can be made with the function tf
H = tf([1], [2, 1])
# which is the same as
H = 1/(1+2*s)
H
[3]:
1
2s + 1
1
[5]: H2
[5]:
1
s2 + s + 1
2 Signals
[10]: # time vector
dt = 0.01
t = np.arange(0, 20+dt, dt)
2
plt.plot(t, y1)
plt.plot(t, y2)
plt.show()
y = np.minimum(t/rampend, 1.)*rampsize
plt.plot(t, y)
plt.show()
3
2.2 Example staircase signal
[13]: t = np.arange(0, 0.5, 0.001)
u = t - np.mod(t, 0.1)
plt.plot(t,u)
plt.show()
4
3 Poles and Zeroes
[14]: # plot poles & zeroes using pzmap(H)
pzmap(H3)
plt.show()
5
[15]: # calculate the poles and zeroes of a transfer function
print("Poles: ", H3.pole(),"\n\nZeros: ", H3.zero())
y1, t = step(H3, t, output=0) # if we have multiple output, you can also specify␣
,→which output you want
y2, t = impulse(H3, t)
6
5 Calculating reponse to an arbitrary input
[17]: # we can use the function lsim to calculate the response to an input
# that is defined in a vector
# suppose our input is a ramp and hold signal and our transferfunction
# is H3, the response can be found by
dt = 0.15
t = np.arange(0, 30+dt, dt)
u = np.minimum(t/10, 1.)*3
plt.plot(t, y)
plt.show()
7
6 Calculating settling time, overshoot, rise time and delay time
[18]: # supposed we have the following system response
y, t = step(H3, t)
yend = y[-1]
tsettle = t[(y <= y[-1]*0.95) | (y >= y[-1]*1.05)][-1]
overshoot = np.max(y)/y[-1]*100 - 100
risetime = t[y >= y[-1]*0.9][0] - t[y <= y[-1]*0.1][-1]
delaytime = t[y <= y[-1]*0.1][-1]
7 State-Space systems
ẋ (t) = Ax (t) + Bu(t)
y = Cx + Du
# parameter values
m = 3.5
b = 9
8
k = 60
# matrices
A = np.array([[ 0, 1], [ -k/m, -b/m]])
B = np.array([[ 0], [ 1/m]])
C = np.array([[ 1, 0]])
D = np.array([[ 0]])
# system
sys = ss(A, B, C, D)
# response
dt = 0.01
t = np.arange(0, 5+dt, dt)
y, t = initial(sys, t, x0)
# plot
plt.plot(t, y)
plt.hlines(0, 0, 5)
plt.show()
9
7.2 Response to a sin function as input to a state-space system
[21]: # supposed we have as input a sin function with frequency w
w = 3.9357078 # rad/s
# remember that if the frequency is in rad/s our sin function looks like
t = np.arange(0.0, 20.01, 0.01)
u = np.sin(w * t)
plt.plot(t, y)
plt.show()
10
sys = ss(A, B, C, D)
s = tf([1,0], [1])
#transferfunctions
hb1 = 1/(Jb*s)
hb2 = 1/s
hp1 = 1/(Jp*s)
hp2 = 1/s
#systems
sys1 = ss(hb1)
sys2 = ss(hb2)
sys3 = k
sys4 = ss(hp1)
sys5 = tf(hp2)
sys6 = b
11
Q = np.array([[1, -3, -6], # u1 = -y3 - y6
[2, 1, 0], # u2 = y2
[3, 2, -5], # u3 = y2 - y5
[4, 3, 6], # u4 = y3 + y6
[5, 4, 0], # u5 = y4
[6, 1, -4]]) # u6 = y1 - y4
inputs = [1]
outputs = [1, 2, 4, 5]
H1
[25]:
0.0025s2 + 1.25e − 05s + 2.5e − 05
s3 + 0.0175s2 + 0.035s − 1.626e − 19
[26]: # coefficients
b0, b1, b2 = 1.0, 0.1, 0.5
a0, a1, a2, a3 = 2.3, 6.3, 3.6 , 1.0
12
H = tf([[h.num[0][0]], [(h*s).num[0][0]]], # h*s
[[h.den[0][0]], [(h*s).den[0][0]]])
H
[26]:
0.5s2 +0.1s+1
" #
s3 +3.6s2 +6.3s+2.3
0.5s3 +0.1s2 +s
s3 +3.6s2 +6.3s+2.3
B = [[1.]
[0.]
[0.]]
D = [[0. ]
[0.5]]
13
7.6 Converting a block diagram to a state-space system using “append and connect”
Consider the block diagrma above; ui and yi represent the input and output to each
block, respectively. Each block is defined as a system. We will convert this into
a state-space system using append and connect.
[29]: # Define s variable
s=tf([1,0],[1])
# fill in variables
K1 = 1
K2 = 2.6
K3 = 2.5
K4 = 0.6
# Define systems; the appended system type will the type of the first system
# given. If the following systems do not have a proper form they will be
# transformed automatically
sys1 = ss(K4 * (s/s)) # Multiply with s/s to 'trick' the control module into␣
,→thinking K4 is a transfer function
sys2 = ss(1/s)
sys3 = ss(1/(s+1))
sys4 = ss(1/s)
sys5 = K1 # Will be transformed by the append function
sys6 = K2
sys7 = K3
14
sys8 = 1 # Add dummy sys8 in order to sum y4 and y5
sys = append(sys1,sys2,sys3,sys4,sys5,sys6,sys7,sys8)
Q = np.array([[1,-6,-7], # u1 = -y6 - y7
[2, 1, 0], # u2 = y1
[3, 2, 0], # u3 = y2
[4, 3, 0], # u4 = y3
[5, 2, 0], # u5 = y2
[6, 3, 0], # u6 = y3
[7, 4, 0], # u7 = y4
[8, 4, 5]]) # u8 = y4 + y5
A = [[ 0. -1.56 -1.5 ]
[ 1. -1. 0. ]
[ 0. 1. 0. ]]
B = [[0.6 1. ]
[0. 0. ]
[0. 0. ]]
C = [[1. 0. 1.]]
D = [[0. 0.]]
# fill in variables
K1 = 1
K2 = 2.6
K3 = 2.5
K4 = 0.6
15
D = np.matrix([[0, 0.0]])
B = [[0.6 1. ]
[0. 0. ]
[0. 0. ]]
C = [[1. 0. 1.]]
D = [[0. 0.]]
16
[9.4, 0, -0.4, -0.6]])
B = np.array([[-0.01, 0.06],
[0, 0],
[-32, 5.4],
[2.6, -7]])
C = np.eye(4)
D = np.zeros((4, 2))
sys1 = ss(A, B, C, D)
# feedback gain
Kd = -0.55
K = np.zeros((2,4))
K[-1, -1] = K1
# New ss system
sys2 = sys1.feedback(K)
sys2
B = [[-1.0e-02 6.0e-02]
[ 0.0e+00 0.0e+00]
[-3.2e+01 5.4e+00]
[ 2.6e+00 -7.0e+00]]
C = [[1. 0. 0. 0.]
[0. 1. 0. 0.]
[0. 0. 1. 0.]
[0. 0. 0. 1.]]
D = [[0. 0.]
[0. 0.]
[0. 0.]
[0. 0.]]
17
[32]: (array([4.07381994e+00, 3.28874827e+00, 3.28874827e+00, 1.08937685e-03]),
array([1. , 0.07983139, 0.07983139, 1. ]),
array([-4.07381994e+00+0.j , -2.62545342e-01+3.27825184j,
-2.62545342e-01-3.27825184j, -1.08937685e-03+0.j ]))
with
18
w = np.sqrt(13)
T = 1.4
sisotool(-Heq) # note that we take -Heq, will later correct for that
Now simply click on the blue line to move the purple square to a value of -0.4,
then read off the value for the gain, in this case 0.438. Now we correct for the
minus sign (sisotool(-Heq)) and thus Kθ = −0.438
To find the damping coefficient we can either click on one of the purple squares
for the imaginary poles and read of the value for ζ sp , or we fill in our value for
Kθ and use the damp function.
[34]: K0 = -0.438
Heq = (K0 * H).feedback(1) # NOW WE TAKE THE FEEDBACK LOOP! Since we want to␣
,→know the damping coefficient of the entire system
damp(Heq)
19
-1.8 +4.8j 0.3511 5.126
-1.8 -4.8j 0.3511 5.126
-0.4001 1 0.4001
9 Bode
$ $To convert the magnitude in decibels Ldb to | H(jω) | (unitless):
| H ( jω )| = 10Ldb /20
[35]: # The following functions help to find phase and gain margins as well as other␣
,→useful parameters
gm, pm, wg, wp = margin(Heq) # gain margin, phase margin and respective omega␣
,→values ## NON dB
mag, phase, omega = bode(Heq) # Plots + lists for magnitude, phase and omega ##␣
,→NON dB
gm, pm
20
[36]: # To use the bode plot use
bode([Heq], omega=np.logspace(-5,1,500)) # adjust the range and # of steps as␣
,→pleased
# or
sisotool(Heq, omega=np.logspace(-5,1,500), Hz= False) # last argument do get␣
,→horixzontal axis in rad/s
21
9.1 Reading plots: Stability, Mimimum phase and Type N systems
Stability of a transfer function or a system can be assessed by looking at its
pzmap and its bode plots.
For a stable system, the poles all have a real part smaller or equal to 0:
Re( pi ) ≤ 0
22
In this case, the system is stable. In addition we can see that one zero has a
positive real part. This implies that the system is non minimim phase. In other
words, there will be some negative overshoot with respect to the final response of
the system.
A system achieves minimum phase when: (zeros = z)
Re(z j ) ≤ 0
For a polar plot, stability is assessed by checking where the bode line crosses the
real axis. Indeed, at
Re( H ( jω )) ≤ 0
when
Im( H ( jω )) = 0
then the transfer function will generate an unstable response.
The type of system can be determined by observing the number of poles in the
origin. A Type 1 system will have 1 pole in the origin implying its transfer
23
function will contain a
1/s
term. This will be displayed in the bode plot as a -90 degree downward shift for
the whole spectrum.
You can determine the system type by looking at the phase when omega = 0. If for
instance the phase = 0 when omega = 0 then the phase bode plot will display an
initial phase of 0. This implies the lack of any 1/sˆn terms meaning there will be
no poles in the origin and the system is thus a Type 0 system.
24
Good luck!
[ ]:
25